@envive-ai/react-hooks 0.2.6 → 0.2.7-arthur-1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (361) hide show
  1. package/dist/{NewOrgConfig-BxSuoP9C.cjs → NewOrgConfig-Bo1seKr6.cjs} +2 -2
  2. package/dist/{NewOrgConfig-BfrGpiGk.js → NewOrgConfig-yptI2imS.js} +2 -2
  3. package/dist/{SystemSettingsContext-B8X_Dvw2.js → SystemSettingsContext-BY1BFgAQ.js} +2 -2
  4. package/dist/{SystemSettingsContext-Bkoiobdv.cjs → SystemSettingsContext-EDpRMMt2.cjs} +2 -2
  5. package/dist/{TrackComponentVisibleEvent-DwfGqNTx.js → TrackComponentVisibleEvent-CXhKOwKQ.js} +2 -2
  6. package/dist/{TrackComponentVisibleEvent-DMuX-byo.cjs → TrackComponentVisibleEvent-CgxCqrIt.cjs} +2 -2
  7. package/dist/amplitudeContext-BItT9HmT.js +1 -0
  8. package/dist/{amplitudeContext-ZTQMvVTV.cjs → amplitudeContext-C8tT74Mi.cjs} +9 -9
  9. package/dist/{amplitudeContext-B73xamNe.d.cts → amplitudeContext-CCVyp5RU.d.cts} +1 -1
  10. package/dist/{amplitudeContext-DOqL2Vn8.js → amplitudeContext-DCk6Va-j.js} +9 -9
  11. package/dist/amplitudeContext-DPtyVv3Q.cjs +0 -0
  12. package/dist/{amplitudeContext-CiO9T9c-.d.ts → amplitudeContext-DcRur97Z.d.ts} +1 -1
  13. package/dist/{api-bHEYmSiT.js → api-BWSsazAG.js} +3 -3
  14. package/dist/{api-BvygKEiX.cjs → api-DeW6rHj3.cjs} +3 -3
  15. package/dist/{app-BQw_-JGl.cjs → app-BbPSHefQ.cjs} +2 -2
  16. package/dist/{app-Aqkm_SlS.js → app-CflxT_xI.js} +2 -2
  17. package/dist/application/models/graphql/index.cjs +4 -3
  18. package/dist/application/models/graphql/index.d.cts +2 -2
  19. package/dist/application/models/graphql/index.d.ts +2 -2
  20. package/dist/application/models/graphql/index.js +3 -3
  21. package/dist/application/models/guards/api/index.cjs +3 -3
  22. package/dist/application/models/guards/api/index.d.cts +2 -1
  23. package/dist/application/models/guards/api/index.d.ts +2 -1
  24. package/dist/application/models/guards/api/index.js +3 -3
  25. package/dist/application/models/guards/utils.cjs +1 -1
  26. package/dist/application/models/guards/utils.d.cts +1 -1
  27. package/dist/application/models/guards/utils.d.ts +1 -1
  28. package/dist/application/models/guards/utils.js +1 -1
  29. package/dist/application/models/index.cjs +9 -9
  30. package/dist/application/models/index.d.cts +11 -10
  31. package/dist/application/models/index.d.ts +11 -10
  32. package/dist/application/models/index.js +8 -8
  33. package/dist/application/models/variantInfo/index.d.cts +1 -1
  34. package/dist/application/models/variantInfo/index.d.ts +1 -1
  35. package/dist/application/utils/index.cjs +15 -15
  36. package/dist/application/utils/index.d.cts +14 -13
  37. package/dist/application/utils/index.d.ts +14 -13
  38. package/dist/application/utils/index.js +15 -15
  39. package/dist/atoms/app/index.cjs +11 -11
  40. package/dist/atoms/app/index.d.cts +19 -18
  41. package/dist/atoms/app/index.d.ts +19 -18
  42. package/dist/atoms/app/index.js +11 -11
  43. package/dist/atoms/chat/index.cjs +16 -16
  44. package/dist/atoms/chat/index.d.cts +39 -38
  45. package/dist/atoms/chat/index.d.ts +39 -38
  46. package/dist/atoms/chat/index.js +16 -16
  47. package/dist/atoms/globalSearch/index.d.cts +5 -5
  48. package/dist/atoms/globalSearch/index.d.ts +6 -6
  49. package/dist/atoms/org/index.d.cts +30 -29
  50. package/dist/atoms/org/index.d.ts +30 -29
  51. package/dist/atoms/search/index.cjs +18 -18
  52. package/dist/atoms/search/index.d.cts +13 -12
  53. package/dist/atoms/search/index.d.ts +13 -12
  54. package/dist/atoms/search/index.js +18 -18
  55. package/dist/atoms/search/types.d.cts +4 -2
  56. package/dist/atoms/search/types.d.ts +4 -2
  57. package/dist/atoms/search/utils.d.ts +1 -1
  58. package/dist/{cdnContext-Cd0Kvt6g.cjs → cdnContext-CaDyQ_5p.cjs} +2 -2
  59. package/dist/{cdnContext-D8pHA9gh.js → cdnContext-CtrIlAqX.js} +2 -2
  60. package/dist/{chat-U1IgKNij.js → chat-BjhQCyW_.js} +8 -8
  61. package/dist/{chat-DwNALtox.cjs → chat-BkPax29G.cjs} +13 -7
  62. package/dist/{chat-EJbfGWRr.js → chat-ClvJ9xEj.js} +1 -1
  63. package/dist/{chat-CJ9D8n7g.cjs → chat-DCGriB7h.cjs} +1 -1
  64. package/dist/{chatElementDisplayLocation-DWmfNX_u.d.cts → chatElementDisplayLocation-B7vr33eG.d.cts} +1 -1
  65. package/dist/{chatElementDisplayLocation-DbmdwAff.d.ts → chatElementDisplayLocation-D4XF0UfI.d.ts} +1 -1
  66. package/dist/{chatSearch-Bb2SMr9X.js → chatSearch-BsYlFvpv.js} +4 -4
  67. package/dist/{chatSearch-DtE2cUQw.cjs → chatSearch-C3N3iIxu.cjs} +4 -4
  68. package/dist/{chatState-B3Dyrd9M.cjs → chatState-CJ52Ag_7.cjs} +3 -3
  69. package/dist/{chatState-BXBN-12W.js → chatState-CXA1vF16.js} +3 -3
  70. package/dist/{commerce-api-Dx02FCQ7.cjs → commerce-api-DA1QGGMK.cjs} +8 -7
  71. package/dist/{commerce-api-DXbnMNT8.js → commerce-api-rgj30eEp.js} +8 -7
  72. package/dist/{common-CuwWqIJ1.cjs → common-DQPvV_S_.cjs} +1 -1
  73. package/dist/{common-Df2bwzd2.js → common-c_4eX0qn.js} +1 -1
  74. package/dist/{components-QGCWJ26c.js → components-CDpaMUjK.js} +1 -1
  75. package/dist/{components-BCfFLf9X.cjs → components-DKwVHIjq.cjs} +1 -1
  76. package/dist/config/index.cjs +4 -4
  77. package/dist/config/index.d.cts +4 -4
  78. package/dist/config/index.d.ts +4 -4
  79. package/dist/config/index.js +4 -4
  80. package/dist/config/locators/components/chat/index.cjs +1 -1
  81. package/dist/config/locators/components/chat/index.d.cts +1 -1
  82. package/dist/config/locators/components/chat/index.d.ts +1 -1
  83. package/dist/config/locators/components/chat/index.js +1 -1
  84. package/dist/config/locators/components/common/index.cjs +1 -1
  85. package/dist/config/locators/components/common/index.d.cts +1 -1
  86. package/dist/config/locators/components/common/index.d.ts +1 -1
  87. package/dist/config/locators/components/common/index.js +1 -1
  88. package/dist/config/locators/components/index.cjs +1 -1
  89. package/dist/config/locators/components/index.d.cts +1 -1
  90. package/dist/config/locators/components/index.d.ts +1 -1
  91. package/dist/config/locators/components/index.js +1 -1
  92. package/dist/config/locators/index.cjs +4 -4
  93. package/dist/config/locators/index.d.cts +4 -4
  94. package/dist/config/locators/index.d.ts +4 -4
  95. package/dist/config/locators/index.js +4 -4
  96. package/dist/contexts/amplitudeContext/index.cjs +15 -14
  97. package/dist/contexts/amplitudeContext/index.d.cts +1 -1
  98. package/dist/contexts/amplitudeContext/index.d.ts +1 -1
  99. package/dist/contexts/amplitudeContext/index.js +15 -14
  100. package/dist/contexts/cdnContext/index.cjs +4 -4
  101. package/dist/contexts/cdnContext/index.js +4 -4
  102. package/dist/contexts/chatContext/index.cjs +23 -23
  103. package/dist/contexts/chatContext/index.d.cts +4 -4
  104. package/dist/contexts/chatContext/index.d.ts +4 -4
  105. package/dist/contexts/chatContext/index.js +23 -23
  106. package/dist/contexts/enviveConfigContext/index.cjs +4 -4
  107. package/dist/contexts/enviveConfigContext/index.d.cts +3 -3
  108. package/dist/contexts/enviveConfigContext/index.d.ts +3 -3
  109. package/dist/contexts/enviveConfigContext/index.js +4 -4
  110. package/dist/contexts/enviveCssContext/index.cjs +15 -15
  111. package/dist/contexts/enviveCssContext/index.js +15 -15
  112. package/dist/contexts/featureFlagContext/index.cjs +6 -6
  113. package/dist/contexts/featureFlagContext/index.d.cts +3 -3
  114. package/dist/contexts/featureFlagContext/index.d.ts +3 -3
  115. package/dist/contexts/featureFlagContext/index.js +6 -6
  116. package/dist/contexts/featureFlagServiceContext/index.cjs +3 -3
  117. package/dist/contexts/featureFlagServiceContext/index.d.cts +4 -4
  118. package/dist/contexts/featureFlagServiceContext/index.d.ts +4 -4
  119. package/dist/contexts/featureFlagServiceContext/index.js +3 -3
  120. package/dist/contexts/graphqlContext/index.cjs +11 -11
  121. package/dist/contexts/graphqlContext/index.d.cts +13 -12
  122. package/dist/contexts/graphqlContext/index.d.ts +13 -12
  123. package/dist/contexts/graphqlContext/index.js +11 -11
  124. package/dist/contexts/localStorageContext/index.cjs +2 -2
  125. package/dist/contexts/localStorageContext/index.js +2 -2
  126. package/dist/contexts/newOrgConfigContext/index.cjs +14 -14
  127. package/dist/contexts/newOrgConfigContext/index.d.cts +14 -13
  128. package/dist/contexts/newOrgConfigContext/index.d.ts +14 -13
  129. package/dist/contexts/newOrgConfigContext/index.js +14 -14
  130. package/dist/contexts/searchContext/index.cjs +17 -17
  131. package/dist/contexts/searchContext/index.d.cts +1 -1
  132. package/dist/contexts/searchContext/index.d.ts +1 -1
  133. package/dist/contexts/searchContext/index.js +17 -17
  134. package/dist/contexts/sessionStorageContext/index.cjs +2 -2
  135. package/dist/contexts/sessionStorageContext/index.js +2 -2
  136. package/dist/contexts/shopifyUrlContext/index.cjs +2 -2
  137. package/dist/contexts/shopifyUrlContext/index.d.cts +3 -3
  138. package/dist/contexts/shopifyUrlContext/index.d.ts +3 -3
  139. package/dist/contexts/shopifyUrlContext/index.js +2 -2
  140. package/dist/contexts/systemSettingsContext/index.cjs +4 -4
  141. package/dist/contexts/systemSettingsContext/index.d.cts +16 -15
  142. package/dist/contexts/systemSettingsContext/index.d.ts +16 -15
  143. package/dist/contexts/systemSettingsContext/index.js +4 -4
  144. package/dist/contexts/types.cjs +1 -1
  145. package/dist/contexts/types.d.cts +3 -3
  146. package/dist/contexts/types.d.ts +3 -3
  147. package/dist/contexts/types.js +1 -1
  148. package/dist/contexts/userIdentityContext/index.cjs +18 -18
  149. package/dist/contexts/userIdentityContext/index.d.cts +1 -1
  150. package/dist/contexts/userIdentityContext/index.d.ts +1 -1
  151. package/dist/contexts/userIdentityContext/index.js +18 -18
  152. package/dist/{enviveConfig-BlIkxiAF.js → enviveConfig-DZBohDpc.js} +3 -3
  153. package/dist/{enviveConfig-B42OJ8bK.cjs → enviveConfig-Dv9-esGV.cjs} +3 -3
  154. package/dist/{enviveConfigContext-Y1ahEAMe.cjs → enviveConfigContext-D2OELZDR.cjs} +3 -3
  155. package/dist/{enviveConfigContext-1_EivtCa.js → enviveConfigContext-DrDjCems.js} +3 -3
  156. package/dist/{featureFlagServiceContext-D3Ge8GH5.cjs → featureFlagServiceContext-CJyYItqu.cjs} +3 -3
  157. package/dist/{featureFlagServiceContext-C5U0bshi.d.ts → featureFlagServiceContext-CpxlOkI9.d.ts} +2 -2
  158. package/dist/{featureFlagServiceContext-CAPrb4e_.js → featureFlagServiceContext-FBM6DdMJ.js} +3 -3
  159. package/dist/{featureFlagServiceContext-CiKWV306.d.cts → featureFlagServiceContext-p5UBwPM3.d.cts} +2 -2
  160. package/dist/{featureGates-D4Me_IZH.js → featureGates-KEwAL8p_.js} +1 -1
  161. package/dist/{featureGates-Bt_Y3kZ7.cjs → featureGates-qU_ulhpC.cjs} +1 -1
  162. package/dist/{frontendConfig-BiD1-j48.d.ts → frontendConfig-Cawh5iqv.d.ts} +6 -3
  163. package/dist/{frontendConfig-tVg0hsWZ.d.cts → frontendConfig-iZipB5FG.d.cts} +6 -3
  164. package/dist/graphql-CkxgqsXP.js +48 -0
  165. package/dist/graphql-i3dtpVTl.cjs +71 -0
  166. package/dist/{graphqlContext-Bf3E-V2T.d.cts → graphqlContext-BeyKU1Dr.d.cts} +2 -2
  167. package/dist/{graphqlContext-CDeKzb46.d.ts → graphqlContext-CXQl0hq2.d.ts} +2 -2
  168. package/dist/{graphqlContext-CpwAvnro.cjs → graphqlContext-DP8T3-Kd.cjs} +6 -6
  169. package/dist/{graphqlContext-dyWNSWNv.js → graphqlContext-czH0kIZg.js} +5 -5
  170. package/dist/hooks/AmplitudeOperations/index.cjs +15 -15
  171. package/dist/hooks/AmplitudeOperations/index.d.cts +1 -1
  172. package/dist/hooks/AmplitudeOperations/index.d.ts +1 -1
  173. package/dist/hooks/AmplitudeOperations/index.js +15 -15
  174. package/dist/hooks/AppDetails/index.cjs +14 -14
  175. package/dist/hooks/AppDetails/index.d.cts +12 -11
  176. package/dist/hooks/AppDetails/index.d.ts +12 -11
  177. package/dist/hooks/AppDetails/index.js +14 -14
  178. package/dist/hooks/CdnOperations/index.cjs +4 -4
  179. package/dist/hooks/CdnOperations/index.js +4 -4
  180. package/dist/hooks/ChatToggle/index.cjs +16 -16
  181. package/dist/hooks/ChatToggle/index.d.cts +1 -1
  182. package/dist/hooks/ChatToggle/index.d.ts +1 -1
  183. package/dist/hooks/ChatToggle/index.js +16 -16
  184. package/dist/hooks/ChatToggleAnalytics/index.cjs +16 -16
  185. package/dist/hooks/ChatToggleAnalytics/index.d.cts +1 -1
  186. package/dist/hooks/ChatToggleAnalytics/index.d.ts +1 -1
  187. package/dist/hooks/ChatToggleAnalytics/index.js +16 -16
  188. package/dist/hooks/CustomerSupportHandoff/index.cjs +1 -1
  189. package/dist/hooks/CustomerSupportHandoff/index.js +1 -1
  190. package/dist/hooks/Debounce/index.cjs +20 -2
  191. package/dist/hooks/Debounce/index.js +18 -2
  192. package/dist/hooks/ElementObserver/index.d.cts +1 -1
  193. package/dist/hooks/ElementObserver/index.d.ts +1 -1
  194. package/dist/hooks/GrabAndScroll/index.d.cts +2 -2
  195. package/dist/hooks/GraphQLConfig/index.cjs +12 -12
  196. package/dist/hooks/GraphQLConfig/index.d.cts +13 -12
  197. package/dist/hooks/GraphQLConfig/index.d.ts +13 -12
  198. package/dist/hooks/GraphQLConfig/index.js +12 -12
  199. package/dist/hooks/IdentifyUser/index.cjs +18 -18
  200. package/dist/hooks/IdentifyUser/index.js +18 -18
  201. package/dist/hooks/ImageResolver/index.cjs +10 -10
  202. package/dist/hooks/ImageResolver/index.js +10 -10
  203. package/dist/hooks/LocalStorageOperations/index.cjs +2 -2
  204. package/dist/hooks/LocalStorageOperations/index.js +2 -2
  205. package/dist/hooks/MessageFilter/index.cjs +8 -8
  206. package/dist/hooks/MessageFilter/index.d.cts +12 -11
  207. package/dist/hooks/MessageFilter/index.d.ts +12 -11
  208. package/dist/hooks/MessageFilter/index.js +8 -8
  209. package/dist/hooks/NewOrgConfig/index.cjs +15 -15
  210. package/dist/hooks/NewOrgConfig/index.d.cts +14 -13
  211. package/dist/hooks/NewOrgConfig/index.d.ts +14 -13
  212. package/dist/hooks/NewOrgConfig/index.js +15 -15
  213. package/dist/hooks/Search/index.cjs +1459 -150
  214. package/dist/hooks/Search/index.d.cts +21 -17
  215. package/dist/hooks/Search/index.d.ts +21 -17
  216. package/dist/hooks/Search/index.js +1460 -151
  217. package/dist/hooks/SearchOperations/index.cjs +17 -17
  218. package/dist/hooks/SearchOperations/index.d.cts +1 -1
  219. package/dist/hooks/SearchOperations/index.d.ts +1 -1
  220. package/dist/hooks/SearchOperations/index.js +17 -17
  221. package/dist/hooks/SessionStorageOperations/index.cjs +2 -2
  222. package/dist/hooks/SessionStorageOperations/index.js +2 -2
  223. package/dist/hooks/ShopifyUrlOperations/index.cjs +2 -2
  224. package/dist/hooks/ShopifyUrlOperations/index.d.cts +5 -5
  225. package/dist/hooks/ShopifyUrlOperations/index.d.ts +5 -5
  226. package/dist/hooks/ShopifyUrlOperations/index.js +2 -2
  227. package/dist/hooks/SystemSettingsContext/index.cjs +5 -5
  228. package/dist/hooks/SystemSettingsContext/index.d.cts +14 -13
  229. package/dist/hooks/SystemSettingsContext/index.d.ts +14 -13
  230. package/dist/hooks/SystemSettingsContext/index.js +5 -5
  231. package/dist/hooks/TrackComponentVisibleEvent/index.cjs +15 -15
  232. package/dist/hooks/TrackComponentVisibleEvent/index.d.cts +2 -2
  233. package/dist/hooks/TrackComponentVisibleEvent/index.d.ts +2 -2
  234. package/dist/hooks/TrackComponentVisibleEvent/index.js +15 -15
  235. package/dist/hooks/UpdateAnalyticsProps/index.cjs +14 -14
  236. package/dist/hooks/UpdateAnalyticsProps/index.js +14 -14
  237. package/dist/hooks/utils.d.cts +12 -11
  238. package/dist/hooks/utils.d.ts +12 -11
  239. package/dist/{index-B6xpW8RG.d.ts → index--9_c4tze.d.ts} +1 -1
  240. package/dist/index-BEpDGqnz.d.cts +41 -0
  241. package/dist/{index-D31m6bPU.d.ts → index-BKvFVPUX.d.ts} +1 -1
  242. package/dist/{index-VwfWqyR_.d.ts → index-BNHIIgYk.d.ts} +1 -1
  243. package/dist/{index-Bq39XSmY.d.cts → index-BSd8767K.d.cts} +34 -34
  244. package/dist/index-BUDrAxnl.d.ts +673 -0
  245. package/dist/{index-DzbkQtaK.d.cts → index-CCboEuTO.d.cts} +1 -1
  246. package/dist/{index-FQjyuD3D.d.cts → index-COXkY78t.d.cts} +1 -1
  247. package/dist/{index-wFHfdr6p.d.ts → index-CUO68KG3.d.ts} +34 -34
  248. package/dist/{index--uHjE7L8.d.ts → index-D7htGSQC.d.ts} +1 -1
  249. package/dist/index-DM_5fh8c.d.ts +101 -0
  250. package/dist/index-DU7uw0ba.d.cts +101 -0
  251. package/dist/{index-Da1s8h5C.d.cts → index-DZtnHhlr.d.cts} +1 -1
  252. package/dist/{index-DSRs6N6J.d.ts → index-Dtw-hJdt.d.ts} +1 -1
  253. package/dist/index-Dy3TTIOm.d.cts +673 -0
  254. package/dist/index-ErVcwUnR.d.ts +41 -0
  255. package/dist/{index-Cyq5HiC0.d.cts → index-OkKEOL6H.d.cts} +1 -1
  256. package/dist/{index-9NE86em3.d.cts → index-hAqp0oYb.d.cts} +1 -1
  257. package/dist/interceptors/index.d.cts +13 -12
  258. package/dist/interceptors/index.d.ts +13 -12
  259. package/dist/interceptors/types.d.cts +12 -11
  260. package/dist/interceptors/types.d.ts +12 -11
  261. package/dist/{localStorageContext-DAOJ4be4.js → localStorageContext-BPZ82q-G.js} +2 -2
  262. package/dist/{localStorageContext-C5giszHn.cjs → localStorageContext-NRP-CdmF.cjs} +2 -2
  263. package/dist/{locators-C2fWd-74.js → locators-BMQGmGLq.js} +1 -1
  264. package/dist/{locators-Cx3q6Z_h.cjs → locators-DxYdak1F.cjs} +1 -1
  265. package/dist/{logger-0D_8Ip2L.cjs → logger-TBIl4uIH.cjs} +1 -1
  266. package/dist/{logger-Co0IA3k5.js → logger-W3lqg-4b.js} +1 -1
  267. package/dist/models-CWOgrLCm.js +1284 -0
  268. package/dist/models-DqdLOi2I.cjs +1519 -0
  269. package/dist/{newOrgConfigContext-BMvcqPzD.cjs → newOrgConfigContext-Bet9CgKP.cjs} +4 -4
  270. package/dist/{newOrgConfigContext-6mlrvr1w.js → newOrgConfigContext-Bi_dBNe5.js} +4 -4
  271. package/dist/{newOrgConfigContext-DOdUxlOE.d.cts → newOrgConfigContext-CJI3tsVV.d.cts} +2 -2
  272. package/dist/{newOrgConfigContext-BVyJExeW.d.ts → newOrgConfigContext-I2qceBB4.d.ts} +2 -2
  273. package/dist/{nodeSelector-DYhDUi7v.d.ts → nodeSelector-B5NfnUHv.d.ts} +1 -1
  274. package/dist/{nodeSelector-B3bPtEjX.d.cts → nodeSelector-vKB44CDB.d.cts} +1 -1
  275. package/dist/responseGenerics-D9bS-Dd6.d.ts +148 -0
  276. package/dist/{index-Cx9e-fRi.d.ts → responseGenerics-DWLV09cQ.d.cts} +4 -40
  277. package/dist/{search-y-ioX5Mz.d.cts → search-6RrxBXD6.d.cts} +1 -1
  278. package/dist/{search-D-UfTjB7.cjs → search-Csh2n66W.cjs} +2 -2
  279. package/dist/{search-yawhMv22.js → search-DkiqkogN.js} +2 -2
  280. package/dist/{search-B1OtJe8Z.d.ts → search-DrJiCT7d.d.ts} +1 -1
  281. package/dist/{search-filter-types-BxaNSLs_.d.cts → search-filter-types-BItKtezf.d.cts} +1 -1
  282. package/dist/{search-filter-types-OI9zH3E_.d.ts → search-filter-types-CGFhksO3.d.ts} +1 -1
  283. package/dist/{searchContext-CiqOqTJL.js → searchContext-BmgoAFMF.js} +6 -6
  284. package/dist/{searchContext-CFuwIIW-.cjs → searchContext-DksJfC1s.cjs} +6 -6
  285. package/dist/{sessionStorageContext-BmCW091C.cjs → sessionStorageContext-B6FsNKjj.cjs} +2 -2
  286. package/dist/{sessionStorageContext-CNxkqJi1.js → sessionStorageContext-CLYCm83p.js} +2 -2
  287. package/dist/{shopifyUrlContext-BXbI0PIG.js → shopifyUrlContext-C-PkSgNC.js} +2 -2
  288. package/dist/{shopifyUrlContext-aZMwCfbJ.cjs → shopifyUrlContext-ZOcARiMR.cjs} +2 -2
  289. package/dist/{spiffyWidgets-CvEJIuJx.d.ts → spiffyWidgets-CR6F7FRE.d.ts} +1 -1
  290. package/dist/{spiffyWidgets-BuS00VaQ.d.cts → spiffyWidgets-eNbU1gMc.d.cts} +1 -1
  291. package/dist/{systemSettingsContext-kz6yyiFF.js → systemSettingsContext-DF0jSq9m.js} +2 -2
  292. package/dist/{systemSettingsContext-SozpUezn.cjs → systemSettingsContext-dmE1v6w8.cjs} +2 -2
  293. package/dist/{test-types-DRhqHXw-.d.ts → test-types-BEml_bm3.d.ts} +1 -1
  294. package/dist/{test-types-ThQiO_cc.d.cts → test-types-Dsu8RJZu.d.cts} +1 -1
  295. package/dist/types/index.d.cts +2 -2
  296. package/dist/types/index.d.ts +2 -2
  297. package/dist/{types-r0Z6CwyF.d.ts → types-4LQ7LUCk.d.ts} +2 -2
  298. package/dist/types-BegmH0S1.d.ts +60 -0
  299. package/dist/types-BuvXXGxE.cjs +1 -1
  300. package/dist/types-CtUb63bt.js +76 -0
  301. package/dist/{types-zQGBI-Yo.d.cts → types-DFsSqmWx.d.cts} +2 -2
  302. package/dist/types-DWorwfS-.d.cts +60 -0
  303. package/dist/types-DXnG1tV0.js +1 -1
  304. package/dist/types-UUvB6h05.cjs +106 -0
  305. package/dist/types.d.cts +3 -3
  306. package/dist/types.d.ts +3 -3
  307. package/dist/{unsupportedProductExceptions-B4f9aLjr.cjs → unsupportedProductExceptions-DGENUnEA.cjs} +1 -1
  308. package/dist/{unsupportedProductExceptions-DlmraJm8.js → unsupportedProductExceptions-uQuuelOs.js} +1 -1
  309. package/dist/{urlsParser-DhcEZLc_.cjs → urlsParser-COzMdJaX.cjs} +1 -1
  310. package/dist/{urlsParser-DLCzibqU.js → urlsParser-DxjoLj98.js} +1 -1
  311. package/dist/{useAmplitudeOperations-D0nvIYlt.cjs → useAmplitudeOperations-Bo6YNbTV.cjs} +2 -2
  312. package/dist/{useAmplitudeOperations-Dhks3PgF.js → useAmplitudeOperations-zIRSqmMW.js} +2 -2
  313. package/dist/{useAppDetails-qyaQIbWE.js → useAppDetails-B584gkCs.js} +4 -4
  314. package/dist/{useAppDetails-B9sGmpJ3.cjs → useAppDetails-DczgqeLG.cjs} +4 -4
  315. package/dist/{useGraphQLConfig-BccQUaeW.js → useGraphQLConfig-7UxACM4n.js} +2 -2
  316. package/dist/{useGraphQLConfig-CgKEfVYc.cjs → useGraphQLConfig-D_rF2Sun.cjs} +2 -2
  317. package/dist/{userIdentityContext-Cb6lLv6t.cjs → userIdentityContext-BqbNu7xu.cjs} +5 -5
  318. package/dist/{userIdentityContext-O_DHHPTN.js → userIdentityContext-BxFH9FNQ.js} +5 -5
  319. package/dist/{userIdentityContext-BKLedN4R.d.ts → userIdentityContext-C6kApbuk.d.ts} +1 -1
  320. package/dist/{userIdentityContext-D2oFVFzo.d.cts → userIdentityContext-kU1PIo8K.d.cts} +1 -1
  321. package/dist/{utils-BhyZiDrE.d.ts → utils-BRkGP1eb.d.ts} +1 -1
  322. package/dist/{utils-DFPt3FSw.js → utils-C1ErYSoW.js} +4 -4
  323. package/dist/{utils-B7PAzB_M.d.cts → utils-CBD4g2Nc.d.cts} +4 -4
  324. package/dist/{utils-C6imnLBo.cjs → utils-CDw74BCO.cjs} +1 -1
  325. package/dist/{utils-B3x_9JTY.d.cts → utils-DCrwX6FT.d.cts} +1 -1
  326. package/dist/{utils-C4ci_t0-.js → utils-DIvMgPe8.js} +1 -1
  327. package/dist/{utils-B1v0iXs3.d.ts → utils-QKFAbPT6.d.ts} +4 -4
  328. package/dist/{utils-Cazq8Q3q.cjs → utils-mqfncrhI.cjs} +4 -4
  329. package/package.json +5 -1
  330. package/src/application/commerce-api.ts +2 -0
  331. package/src/application/models/graphql/index.ts +1 -0
  332. package/src/atoms/search/productRetrievalAPI.ts +70 -0
  333. package/src/atoms/search/productRetrievalAdapter.ts +25 -0
  334. package/src/atoms/search/types.ts +13 -0
  335. package/src/contexts/graphqlContext/__tests__/graphqlContext.test.tsx +790 -0
  336. package/src/contexts/localStorageContext/__tests__/localStorageContext.test.tsx +775 -0
  337. package/src/contexts/newOrgConfigContext/__tests__/newOrgConfigContext.test.tsx +495 -0
  338. package/src/contexts/searchContext/__tests__/searchContext.test.tsx +806 -0
  339. package/src/contexts/systemSettingsContext/__tests__/systemSettingsContext.test.tsx +506 -0
  340. package/src/contexts/types.ts +3 -0
  341. package/src/hooks/Search/useRecommendedProducts.ts +48 -0
  342. package/src/hooks/Search/useSearch.tsx +89 -182
  343. package/src/hooks/Search/useSearchInput.ts +263 -0
  344. package/src/types/FilterAttribute.ts +35 -0
  345. package/dist/graphql-OkhsP4ir.js +0 -36
  346. package/dist/graphql-l4dQrsA6.cjs +0 -53
  347. package/dist/index-Bmub8e38.d.cts +0 -98
  348. package/dist/index-CG3P8xE1.d.cts +0 -676
  349. package/dist/index-CiWEYzXl.d.cts +0 -184
  350. package/dist/index-D2VaMPA3.d.ts +0 -98
  351. package/dist/index-fUsw_Mea.d.ts +0 -676
  352. package/dist/models-CkJ-wg9Q.cjs +0 -1537
  353. package/dist/models-UHOY0ak5.js +0 -1296
  354. package/dist/types-BwNrLPSZ.cjs +0 -106
  355. package/dist/types-CKMMb_gX.d.cts +0 -51
  356. package/dist/types-D3uOF0Oy.js +0 -76
  357. package/dist/types-DZPuBnHe.d.ts +0 -51
  358. package/dist/useDebounce-BZDtUAQ8.cjs +0 -26
  359. package/dist/useDebounce-ueblXZI-.js +0 -19
  360. /package/dist/{index-RcVcRKH7.d.cts → index-A0HvA68Y.d.cts} +0 -0
  361. /package/dist/{index-mHc9_XC3.d.ts → index-DNVvRcKu.d.ts} +0 -0
@@ -1,47 +1,1431 @@
1
1
  const require_chunk = require('../../chunk-CUT6urMc.cjs');
2
- require('../../types-BwNrLPSZ.cjs');
2
+ require('../../types-UUvB6h05.cjs');
3
3
  require('../../events-DYY4l817.cjs');
4
4
  const require_utils = require('../../utils-hYTjy7hJ.cjs');
5
- const require_logger = require('../../logger-0D_8Ip2L.cjs');
6
- const require_models = require('../../models-CkJ-wg9Q.cjs');
7
- require('../../featureGates-Bt_Y3kZ7.cjs');
8
- require('../../urlsParser-DhcEZLc_.cjs');
9
- require('../../graphql-l4dQrsA6.cjs');
10
- require('../../utils-C6imnLBo.cjs');
11
- require('../../api-BvygKEiX.cjs');
5
+ const require_models = require('../../models-DqdLOi2I.cjs');
6
+ const require_featureGates = require('../../featureGates-qU_ulhpC.cjs');
7
+ require('../../urlsParser-COzMdJaX.cjs');
8
+ require('../../graphql-i3dtpVTl.cjs');
9
+ require('../../utils-CDw74BCO.cjs');
10
+ require('../../api-DeW6rHj3.cjs');
11
+ require('../../logger-TBIl4uIH.cjs');
12
12
  require('../../utilityTypes-BGbL2WTP.cjs');
13
13
  require('../../variantInfo-DpLn4nHz.cjs');
14
- require('../../localStorageContext-C5giszHn.cjs');
15
- const require_enviveConfig = require('../../enviveConfig-B42OJ8bK.cjs');
14
+ require('../../localStorageContext-NRP-CdmF.cjs');
15
+ const require_enviveConfig = require('../../enviveConfig-Dv9-esGV.cjs');
16
16
  require('../../orgAnalyticsConfig-Bm23fw4s.cjs');
17
17
  require('../../atomStore-CmZbgQHc.cjs');
18
- require('../../app-BQw_-JGl.cjs');
19
- require('../../enviveConfigContext-Y1ahEAMe.cjs');
20
- require('../../featureFlagServiceContext-D3Ge8GH5.cjs');
21
- const require_amplitudeContext = require('../../amplitudeContext-ZTQMvVTV.cjs');
22
- require('../../chatState-B3Dyrd9M.cjs');
18
+ require('../../app-BbPSHefQ.cjs');
19
+ require('../../enviveConfigContext-D2OELZDR.cjs');
20
+ const require_featureFlagServiceContext = require('../../featureFlagServiceContext-CJyYItqu.cjs');
21
+ const require_amplitudeContext = require('../../amplitudeContext-C8tT74Mi.cjs');
22
+ require('../../chatState-CJ52Ag_7.cjs');
23
23
  require('../../graphqlConfig-BnfE0ql5.cjs');
24
- require('../../chat-DwNALtox.cjs');
24
+ const require_chat = require('../../chat-BkPax29G.cjs');
25
25
  const require_globalSearch = require('../../globalSearch-BOG3wmck.cjs');
26
26
  require('../../org-B_cWn2bt.cjs');
27
27
  require('../../newOrgConfigAtom-CPA6Gp6n.cjs');
28
28
  const require_utils$2 = require('../../utils-CvLmSsUj.cjs');
29
- require('../../chatSearch-DtE2cUQw.cjs');
29
+ require('../../chatSearch-C3N3iIxu.cjs');
30
30
  const require_types$1 = require('../../types-BuvXXGxE.cjs');
31
31
  require('../../searchServiceAdapter-BGlvoZFE.cjs');
32
- const require_search = require('../../search-D-UfTjB7.cjs');
33
- require('../../graphqlContext-CpwAvnro.cjs');
34
- require('../../useGraphQLConfig-CgKEfVYc.cjs');
35
- require('../../newOrgConfigContext-BMvcqPzD.cjs');
36
- const require_NewOrgConfig = require('../../NewOrgConfig-BxSuoP9C.cjs');
37
- const require_useDebounce = require('../../useDebounce-BZDtUAQ8.cjs');
32
+ const require_search = require('../../search-Csh2n66W.cjs');
33
+ require('../../amplitudeContext-DPtyVv3Q.cjs');
34
+ require('../../graphqlContext-DP8T3-Kd.cjs');
35
+ require('../../useGraphQLConfig-D_rF2Sun.cjs');
36
+ require('../../newOrgConfigContext-Bet9CgKP.cjs');
37
+ const require_NewOrgConfig = require('../../NewOrgConfig-Bo1seKr6.cjs');
38
38
  require('../../useIntersection-DSDREfj6.cjs');
39
- const require_TrackComponentVisibleEvent = require('../../TrackComponentVisibleEvent-DMuX-byo.cjs');
39
+ const require_TrackComponentVisibleEvent = require('../../TrackComponentVisibleEvent-CgxCqrIt.cjs');
40
40
  let react = require("react");
41
41
  react = require_chunk.__toESM(react);
42
42
  let jotai = require("jotai");
43
43
  jotai = require_chunk.__toESM(jotai);
44
44
 
45
+ //#region node_modules/fuse.js/dist/fuse.mjs
46
+ /**
47
+ * Fuse.js v7.1.0 - Lightweight fuzzy-search (http://fusejs.io)
48
+ *
49
+ * Copyright (c) 2025 Kiro Risk (http://kiro.me)
50
+ * All Rights Reserved. Apache Software License 2.0
51
+ *
52
+ * http://www.apache.org/licenses/LICENSE-2.0
53
+ */
54
+ function isArray(value) {
55
+ return !Array.isArray ? getTag(value) === "[object Array]" : Array.isArray(value);
56
+ }
57
+ const INFINITY = Infinity;
58
+ function baseToString(value) {
59
+ if (typeof value == "string") return value;
60
+ let result = value + "";
61
+ return result == "0" && 1 / value == -INFINITY ? "-0" : result;
62
+ }
63
+ function toString(value) {
64
+ return value == null ? "" : baseToString(value);
65
+ }
66
+ function isString(value) {
67
+ return typeof value === "string";
68
+ }
69
+ function isNumber(value) {
70
+ return typeof value === "number";
71
+ }
72
+ function isBoolean(value) {
73
+ return value === true || value === false || isObjectLike(value) && getTag(value) == "[object Boolean]";
74
+ }
75
+ function isObject(value) {
76
+ return typeof value === "object";
77
+ }
78
+ function isObjectLike(value) {
79
+ return isObject(value) && value !== null;
80
+ }
81
+ function isDefined(value) {
82
+ return value !== void 0 && value !== null;
83
+ }
84
+ function isBlank(value) {
85
+ return !value.trim().length;
86
+ }
87
+ function getTag(value) {
88
+ return value == null ? value === void 0 ? "[object Undefined]" : "[object Null]" : Object.prototype.toString.call(value);
89
+ }
90
+ const INCORRECT_INDEX_TYPE = "Incorrect 'index' type";
91
+ const LOGICAL_SEARCH_INVALID_QUERY_FOR_KEY = (key) => `Invalid value for key ${key}`;
92
+ const PATTERN_LENGTH_TOO_LARGE = (max) => `Pattern length exceeds max of ${max}.`;
93
+ const MISSING_KEY_PROPERTY = (name) => `Missing ${name} property in key`;
94
+ const INVALID_KEY_WEIGHT_VALUE = (key) => `Property 'weight' in key '${key}' must be a positive integer`;
95
+ const hasOwn = Object.prototype.hasOwnProperty;
96
+ var KeyStore = class {
97
+ constructor(keys) {
98
+ this._keys = [];
99
+ this._keyMap = {};
100
+ let totalWeight = 0;
101
+ keys.forEach((key) => {
102
+ let obj = createKey(key);
103
+ this._keys.push(obj);
104
+ this._keyMap[obj.id] = obj;
105
+ totalWeight += obj.weight;
106
+ });
107
+ this._keys.forEach((key) => {
108
+ key.weight /= totalWeight;
109
+ });
110
+ }
111
+ get(keyId) {
112
+ return this._keyMap[keyId];
113
+ }
114
+ keys() {
115
+ return this._keys;
116
+ }
117
+ toJSON() {
118
+ return JSON.stringify(this._keys);
119
+ }
120
+ };
121
+ function createKey(key) {
122
+ let path = null;
123
+ let id = null;
124
+ let src = null;
125
+ let weight = 1;
126
+ let getFn = null;
127
+ if (isString(key) || isArray(key)) {
128
+ src = key;
129
+ path = createKeyPath(key);
130
+ id = createKeyId(key);
131
+ } else {
132
+ if (!hasOwn.call(key, "name")) throw new Error(MISSING_KEY_PROPERTY("name"));
133
+ const name = key.name;
134
+ src = name;
135
+ if (hasOwn.call(key, "weight")) {
136
+ weight = key.weight;
137
+ if (weight <= 0) throw new Error(INVALID_KEY_WEIGHT_VALUE(name));
138
+ }
139
+ path = createKeyPath(name);
140
+ id = createKeyId(name);
141
+ getFn = key.getFn;
142
+ }
143
+ return {
144
+ path,
145
+ id,
146
+ weight,
147
+ src,
148
+ getFn
149
+ };
150
+ }
151
+ function createKeyPath(key) {
152
+ return isArray(key) ? key : key.split(".");
153
+ }
154
+ function createKeyId(key) {
155
+ return isArray(key) ? key.join(".") : key;
156
+ }
157
+ function get(obj, path) {
158
+ let list = [];
159
+ let arr = false;
160
+ const deepGet = (obj$1, path$1, index) => {
161
+ if (!isDefined(obj$1)) return;
162
+ if (!path$1[index]) list.push(obj$1);
163
+ else {
164
+ let key = path$1[index];
165
+ const value = obj$1[key];
166
+ if (!isDefined(value)) return;
167
+ if (index === path$1.length - 1 && (isString(value) || isNumber(value) || isBoolean(value))) list.push(toString(value));
168
+ else if (isArray(value)) {
169
+ arr = true;
170
+ for (let i = 0, len = value.length; i < len; i += 1) deepGet(value[i], path$1, index + 1);
171
+ } else if (path$1.length) deepGet(value, path$1, index + 1);
172
+ }
173
+ };
174
+ deepGet(obj, isString(path) ? path.split(".") : path, 0);
175
+ return arr ? list : list[0];
176
+ }
177
+ const MatchOptions = {
178
+ includeMatches: false,
179
+ findAllMatches: false,
180
+ minMatchCharLength: 1
181
+ };
182
+ const BasicOptions = {
183
+ isCaseSensitive: false,
184
+ ignoreDiacritics: false,
185
+ includeScore: false,
186
+ keys: [],
187
+ shouldSort: true,
188
+ sortFn: (a, b) => a.score === b.score ? a.idx < b.idx ? -1 : 1 : a.score < b.score ? -1 : 1
189
+ };
190
+ const FuzzyOptions = {
191
+ location: 0,
192
+ threshold: .6,
193
+ distance: 100
194
+ };
195
+ const AdvancedOptions = {
196
+ useExtendedSearch: false,
197
+ getFn: get,
198
+ ignoreLocation: false,
199
+ ignoreFieldNorm: false,
200
+ fieldNormWeight: 1
201
+ };
202
+ var Config = {
203
+ ...BasicOptions,
204
+ ...MatchOptions,
205
+ ...FuzzyOptions,
206
+ ...AdvancedOptions
207
+ };
208
+ const SPACE = /[^ ]+/g;
209
+ function norm(weight = 1, mantissa = 3) {
210
+ const cache = /* @__PURE__ */ new Map();
211
+ const m = Math.pow(10, mantissa);
212
+ return {
213
+ get(value) {
214
+ const numTokens = value.match(SPACE).length;
215
+ if (cache.has(numTokens)) return cache.get(numTokens);
216
+ const norm$1 = 1 / Math.pow(numTokens, .5 * weight);
217
+ const n = parseFloat(Math.round(norm$1 * m) / m);
218
+ cache.set(numTokens, n);
219
+ return n;
220
+ },
221
+ clear() {
222
+ cache.clear();
223
+ }
224
+ };
225
+ }
226
+ var FuseIndex = class {
227
+ constructor({ getFn = Config.getFn, fieldNormWeight = Config.fieldNormWeight } = {}) {
228
+ this.norm = norm(fieldNormWeight, 3);
229
+ this.getFn = getFn;
230
+ this.isCreated = false;
231
+ this.setIndexRecords();
232
+ }
233
+ setSources(docs = []) {
234
+ this.docs = docs;
235
+ }
236
+ setIndexRecords(records = []) {
237
+ this.records = records;
238
+ }
239
+ setKeys(keys = []) {
240
+ this.keys = keys;
241
+ this._keysMap = {};
242
+ keys.forEach((key, idx) => {
243
+ this._keysMap[key.id] = idx;
244
+ });
245
+ }
246
+ create() {
247
+ if (this.isCreated || !this.docs.length) return;
248
+ this.isCreated = true;
249
+ if (isString(this.docs[0])) this.docs.forEach((doc, docIndex) => {
250
+ this._addString(doc, docIndex);
251
+ });
252
+ else this.docs.forEach((doc, docIndex) => {
253
+ this._addObject(doc, docIndex);
254
+ });
255
+ this.norm.clear();
256
+ }
257
+ add(doc) {
258
+ const idx = this.size();
259
+ if (isString(doc)) this._addString(doc, idx);
260
+ else this._addObject(doc, idx);
261
+ }
262
+ removeAt(idx) {
263
+ this.records.splice(idx, 1);
264
+ for (let i = idx, len = this.size(); i < len; i += 1) this.records[i].i -= 1;
265
+ }
266
+ getValueForItemAtKeyId(item, keyId) {
267
+ return item[this._keysMap[keyId]];
268
+ }
269
+ size() {
270
+ return this.records.length;
271
+ }
272
+ _addString(doc, docIndex) {
273
+ if (!isDefined(doc) || isBlank(doc)) return;
274
+ let record = {
275
+ v: doc,
276
+ i: docIndex,
277
+ n: this.norm.get(doc)
278
+ };
279
+ this.records.push(record);
280
+ }
281
+ _addObject(doc, docIndex) {
282
+ let record = {
283
+ i: docIndex,
284
+ $: {}
285
+ };
286
+ this.keys.forEach((key, keyIndex) => {
287
+ let value = key.getFn ? key.getFn(doc) : this.getFn(doc, key.path);
288
+ if (!isDefined(value)) return;
289
+ if (isArray(value)) {
290
+ let subRecords = [];
291
+ const stack = [{
292
+ nestedArrIndex: -1,
293
+ value
294
+ }];
295
+ while (stack.length) {
296
+ const { nestedArrIndex, value: value$1 } = stack.pop();
297
+ if (!isDefined(value$1)) continue;
298
+ if (isString(value$1) && !isBlank(value$1)) {
299
+ let subRecord = {
300
+ v: value$1,
301
+ i: nestedArrIndex,
302
+ n: this.norm.get(value$1)
303
+ };
304
+ subRecords.push(subRecord);
305
+ } else if (isArray(value$1)) value$1.forEach((item, k) => {
306
+ stack.push({
307
+ nestedArrIndex: k,
308
+ value: item
309
+ });
310
+ });
311
+ }
312
+ record.$[keyIndex] = subRecords;
313
+ } else if (isString(value) && !isBlank(value)) {
314
+ let subRecord = {
315
+ v: value,
316
+ n: this.norm.get(value)
317
+ };
318
+ record.$[keyIndex] = subRecord;
319
+ }
320
+ });
321
+ this.records.push(record);
322
+ }
323
+ toJSON() {
324
+ return {
325
+ keys: this.keys,
326
+ records: this.records
327
+ };
328
+ }
329
+ };
330
+ function createIndex(keys, docs, { getFn = Config.getFn, fieldNormWeight = Config.fieldNormWeight } = {}) {
331
+ const myIndex = new FuseIndex({
332
+ getFn,
333
+ fieldNormWeight
334
+ });
335
+ myIndex.setKeys(keys.map(createKey));
336
+ myIndex.setSources(docs);
337
+ myIndex.create();
338
+ return myIndex;
339
+ }
340
+ function parseIndex(data, { getFn = Config.getFn, fieldNormWeight = Config.fieldNormWeight } = {}) {
341
+ const { keys, records } = data;
342
+ const myIndex = new FuseIndex({
343
+ getFn,
344
+ fieldNormWeight
345
+ });
346
+ myIndex.setKeys(keys);
347
+ myIndex.setIndexRecords(records);
348
+ return myIndex;
349
+ }
350
+ function computeScore$1(pattern, { errors = 0, currentLocation = 0, expectedLocation = 0, distance = Config.distance, ignoreLocation = Config.ignoreLocation } = {}) {
351
+ const accuracy = errors / pattern.length;
352
+ if (ignoreLocation) return accuracy;
353
+ const proximity = Math.abs(expectedLocation - currentLocation);
354
+ if (!distance) return proximity ? 1 : accuracy;
355
+ return accuracy + proximity / distance;
356
+ }
357
+ function convertMaskToIndices(matchmask = [], minMatchCharLength = Config.minMatchCharLength) {
358
+ let indices = [];
359
+ let start = -1;
360
+ let end = -1;
361
+ let i = 0;
362
+ for (let len = matchmask.length; i < len; i += 1) {
363
+ let match = matchmask[i];
364
+ if (match && start === -1) start = i;
365
+ else if (!match && start !== -1) {
366
+ end = i - 1;
367
+ if (end - start + 1 >= minMatchCharLength) indices.push([start, end]);
368
+ start = -1;
369
+ }
370
+ }
371
+ if (matchmask[i - 1] && i - start >= minMatchCharLength) indices.push([start, i - 1]);
372
+ return indices;
373
+ }
374
+ const MAX_BITS = 32;
375
+ function search(text, pattern, patternAlphabet, { location = Config.location, distance = Config.distance, threshold = Config.threshold, findAllMatches = Config.findAllMatches, minMatchCharLength = Config.minMatchCharLength, includeMatches = Config.includeMatches, ignoreLocation = Config.ignoreLocation } = {}) {
376
+ if (pattern.length > MAX_BITS) throw new Error(PATTERN_LENGTH_TOO_LARGE(MAX_BITS));
377
+ const patternLen = pattern.length;
378
+ const textLen = text.length;
379
+ const expectedLocation = Math.max(0, Math.min(location, textLen));
380
+ let currentThreshold = threshold;
381
+ let bestLocation = expectedLocation;
382
+ const computeMatches = minMatchCharLength > 1 || includeMatches;
383
+ const matchMask = computeMatches ? Array(textLen) : [];
384
+ let index;
385
+ while ((index = text.indexOf(pattern, bestLocation)) > -1) {
386
+ let score = computeScore$1(pattern, {
387
+ currentLocation: index,
388
+ expectedLocation,
389
+ distance,
390
+ ignoreLocation
391
+ });
392
+ currentThreshold = Math.min(score, currentThreshold);
393
+ bestLocation = index + patternLen;
394
+ if (computeMatches) {
395
+ let i = 0;
396
+ while (i < patternLen) {
397
+ matchMask[index + i] = 1;
398
+ i += 1;
399
+ }
400
+ }
401
+ }
402
+ bestLocation = -1;
403
+ let lastBitArr = [];
404
+ let finalScore = 1;
405
+ let binMax = patternLen + textLen;
406
+ const mask = 1 << patternLen - 1;
407
+ for (let i = 0; i < patternLen; i += 1) {
408
+ let binMin = 0;
409
+ let binMid = binMax;
410
+ while (binMin < binMid) {
411
+ if (computeScore$1(pattern, {
412
+ errors: i,
413
+ currentLocation: expectedLocation + binMid,
414
+ expectedLocation,
415
+ distance,
416
+ ignoreLocation
417
+ }) <= currentThreshold) binMin = binMid;
418
+ else binMax = binMid;
419
+ binMid = Math.floor((binMax - binMin) / 2 + binMin);
420
+ }
421
+ binMax = binMid;
422
+ let start = Math.max(1, expectedLocation - binMid + 1);
423
+ let finish = findAllMatches ? textLen : Math.min(expectedLocation + binMid, textLen) + patternLen;
424
+ let bitArr = Array(finish + 2);
425
+ bitArr[finish + 1] = (1 << i) - 1;
426
+ for (let j = finish; j >= start; j -= 1) {
427
+ let currentLocation = j - 1;
428
+ let charMatch = patternAlphabet[text.charAt(currentLocation)];
429
+ if (computeMatches) matchMask[currentLocation] = +!!charMatch;
430
+ bitArr[j] = (bitArr[j + 1] << 1 | 1) & charMatch;
431
+ if (i) bitArr[j] |= (lastBitArr[j + 1] | lastBitArr[j]) << 1 | 1 | lastBitArr[j + 1];
432
+ if (bitArr[j] & mask) {
433
+ finalScore = computeScore$1(pattern, {
434
+ errors: i,
435
+ currentLocation,
436
+ expectedLocation,
437
+ distance,
438
+ ignoreLocation
439
+ });
440
+ if (finalScore <= currentThreshold) {
441
+ currentThreshold = finalScore;
442
+ bestLocation = currentLocation;
443
+ if (bestLocation <= expectedLocation) break;
444
+ start = Math.max(1, 2 * expectedLocation - bestLocation);
445
+ }
446
+ }
447
+ }
448
+ if (computeScore$1(pattern, {
449
+ errors: i + 1,
450
+ currentLocation: expectedLocation,
451
+ expectedLocation,
452
+ distance,
453
+ ignoreLocation
454
+ }) > currentThreshold) break;
455
+ lastBitArr = bitArr;
456
+ }
457
+ const result = {
458
+ isMatch: bestLocation >= 0,
459
+ score: Math.max(.001, finalScore)
460
+ };
461
+ if (computeMatches) {
462
+ const indices = convertMaskToIndices(matchMask, minMatchCharLength);
463
+ if (!indices.length) result.isMatch = false;
464
+ else if (includeMatches) result.indices = indices;
465
+ }
466
+ return result;
467
+ }
468
+ function createPatternAlphabet(pattern) {
469
+ let mask = {};
470
+ for (let i = 0, len = pattern.length; i < len; i += 1) {
471
+ const char = pattern.charAt(i);
472
+ mask[char] = (mask[char] || 0) | 1 << len - i - 1;
473
+ }
474
+ return mask;
475
+ }
476
+ const stripDiacritics = String.prototype.normalize ? ((str) => str.normalize("NFD").replace(/[\u0300-\u036F\u0483-\u0489\u0591-\u05BD\u05BF\u05C1\u05C2\u05C4\u05C5\u05C7\u0610-\u061A\u064B-\u065F\u0670\u06D6-\u06DC\u06DF-\u06E4\u06E7\u06E8\u06EA-\u06ED\u0711\u0730-\u074A\u07A6-\u07B0\u07EB-\u07F3\u07FD\u0816-\u0819\u081B-\u0823\u0825-\u0827\u0829-\u082D\u0859-\u085B\u08D3-\u08E1\u08E3-\u0903\u093A-\u093C\u093E-\u094F\u0951-\u0957\u0962\u0963\u0981-\u0983\u09BC\u09BE-\u09C4\u09C7\u09C8\u09CB-\u09CD\u09D7\u09E2\u09E3\u09FE\u0A01-\u0A03\u0A3C\u0A3E-\u0A42\u0A47\u0A48\u0A4B-\u0A4D\u0A51\u0A70\u0A71\u0A75\u0A81-\u0A83\u0ABC\u0ABE-\u0AC5\u0AC7-\u0AC9\u0ACB-\u0ACD\u0AE2\u0AE3\u0AFA-\u0AFF\u0B01-\u0B03\u0B3C\u0B3E-\u0B44\u0B47\u0B48\u0B4B-\u0B4D\u0B56\u0B57\u0B62\u0B63\u0B82\u0BBE-\u0BC2\u0BC6-\u0BC8\u0BCA-\u0BCD\u0BD7\u0C00-\u0C04\u0C3E-\u0C44\u0C46-\u0C48\u0C4A-\u0C4D\u0C55\u0C56\u0C62\u0C63\u0C81-\u0C83\u0CBC\u0CBE-\u0CC4\u0CC6-\u0CC8\u0CCA-\u0CCD\u0CD5\u0CD6\u0CE2\u0CE3\u0D00-\u0D03\u0D3B\u0D3C\u0D3E-\u0D44\u0D46-\u0D48\u0D4A-\u0D4D\u0D57\u0D62\u0D63\u0D82\u0D83\u0DCA\u0DCF-\u0DD4\u0DD6\u0DD8-\u0DDF\u0DF2\u0DF3\u0E31\u0E34-\u0E3A\u0E47-\u0E4E\u0EB1\u0EB4-\u0EB9\u0EBB\u0EBC\u0EC8-\u0ECD\u0F18\u0F19\u0F35\u0F37\u0F39\u0F3E\u0F3F\u0F71-\u0F84\u0F86\u0F87\u0F8D-\u0F97\u0F99-\u0FBC\u0FC6\u102B-\u103E\u1056-\u1059\u105E-\u1060\u1062-\u1064\u1067-\u106D\u1071-\u1074\u1082-\u108D\u108F\u109A-\u109D\u135D-\u135F\u1712-\u1714\u1732-\u1734\u1752\u1753\u1772\u1773\u17B4-\u17D3\u17DD\u180B-\u180D\u1885\u1886\u18A9\u1920-\u192B\u1930-\u193B\u1A17-\u1A1B\u1A55-\u1A5E\u1A60-\u1A7C\u1A7F\u1AB0-\u1ABE\u1B00-\u1B04\u1B34-\u1B44\u1B6B-\u1B73\u1B80-\u1B82\u1BA1-\u1BAD\u1BE6-\u1BF3\u1C24-\u1C37\u1CD0-\u1CD2\u1CD4-\u1CE8\u1CED\u1CF2-\u1CF4\u1CF7-\u1CF9\u1DC0-\u1DF9\u1DFB-\u1DFF\u20D0-\u20F0\u2CEF-\u2CF1\u2D7F\u2DE0-\u2DFF\u302A-\u302F\u3099\u309A\uA66F-\uA672\uA674-\uA67D\uA69E\uA69F\uA6F0\uA6F1\uA802\uA806\uA80B\uA823-\uA827\uA880\uA881\uA8B4-\uA8C5\uA8E0-\uA8F1\uA8FF\uA926-\uA92D\uA947-\uA953\uA980-\uA983\uA9B3-\uA9C0\uA9E5\uAA29-\uAA36\uAA43\uAA4C\uAA4D\uAA7B-\uAA7D\uAAB0\uAAB2-\uAAB4\uAAB7\uAAB8\uAABE\uAABF\uAAC1\uAAEB-\uAAEF\uAAF5\uAAF6\uABE3-\uABEA\uABEC\uABED\uFB1E\uFE00-\uFE0F\uFE20-\uFE2F]/g, "")) : ((str) => str);
477
+ var BitapSearch = class {
478
+ constructor(pattern, { location = Config.location, threshold = Config.threshold, distance = Config.distance, includeMatches = Config.includeMatches, findAllMatches = Config.findAllMatches, minMatchCharLength = Config.minMatchCharLength, isCaseSensitive = Config.isCaseSensitive, ignoreDiacritics = Config.ignoreDiacritics, ignoreLocation = Config.ignoreLocation } = {}) {
479
+ this.options = {
480
+ location,
481
+ threshold,
482
+ distance,
483
+ includeMatches,
484
+ findAllMatches,
485
+ minMatchCharLength,
486
+ isCaseSensitive,
487
+ ignoreDiacritics,
488
+ ignoreLocation
489
+ };
490
+ pattern = isCaseSensitive ? pattern : pattern.toLowerCase();
491
+ pattern = ignoreDiacritics ? stripDiacritics(pattern) : pattern;
492
+ this.pattern = pattern;
493
+ this.chunks = [];
494
+ if (!this.pattern.length) return;
495
+ const addChunk = (pattern$1, startIndex) => {
496
+ this.chunks.push({
497
+ pattern: pattern$1,
498
+ alphabet: createPatternAlphabet(pattern$1),
499
+ startIndex
500
+ });
501
+ };
502
+ const len = this.pattern.length;
503
+ if (len > MAX_BITS) {
504
+ let i = 0;
505
+ const remainder = len % MAX_BITS;
506
+ const end = len - remainder;
507
+ while (i < end) {
508
+ addChunk(this.pattern.substr(i, MAX_BITS), i);
509
+ i += MAX_BITS;
510
+ }
511
+ if (remainder) {
512
+ const startIndex = len - MAX_BITS;
513
+ addChunk(this.pattern.substr(startIndex), startIndex);
514
+ }
515
+ } else addChunk(this.pattern, 0);
516
+ }
517
+ searchIn(text) {
518
+ const { isCaseSensitive, ignoreDiacritics, includeMatches } = this.options;
519
+ text = isCaseSensitive ? text : text.toLowerCase();
520
+ text = ignoreDiacritics ? stripDiacritics(text) : text;
521
+ if (this.pattern === text) {
522
+ let result$1 = {
523
+ isMatch: true,
524
+ score: 0
525
+ };
526
+ if (includeMatches) result$1.indices = [[0, text.length - 1]];
527
+ return result$1;
528
+ }
529
+ const { location, distance, threshold, findAllMatches, minMatchCharLength, ignoreLocation } = this.options;
530
+ let allIndices = [];
531
+ let totalScore = 0;
532
+ let hasMatches = false;
533
+ this.chunks.forEach(({ pattern, alphabet, startIndex }) => {
534
+ const { isMatch, score, indices } = search(text, pattern, alphabet, {
535
+ location: location + startIndex,
536
+ distance,
537
+ threshold,
538
+ findAllMatches,
539
+ minMatchCharLength,
540
+ includeMatches,
541
+ ignoreLocation
542
+ });
543
+ if (isMatch) hasMatches = true;
544
+ totalScore += score;
545
+ if (isMatch && indices) allIndices = [...allIndices, ...indices];
546
+ });
547
+ let result = {
548
+ isMatch: hasMatches,
549
+ score: hasMatches ? totalScore / this.chunks.length : 1
550
+ };
551
+ if (hasMatches && includeMatches) result.indices = allIndices;
552
+ return result;
553
+ }
554
+ };
555
+ var BaseMatch = class {
556
+ constructor(pattern) {
557
+ this.pattern = pattern;
558
+ }
559
+ static isMultiMatch(pattern) {
560
+ return getMatch(pattern, this.multiRegex);
561
+ }
562
+ static isSingleMatch(pattern) {
563
+ return getMatch(pattern, this.singleRegex);
564
+ }
565
+ search() {}
566
+ };
567
+ function getMatch(pattern, exp) {
568
+ const matches = pattern.match(exp);
569
+ return matches ? matches[1] : null;
570
+ }
571
+ var ExactMatch = class extends BaseMatch {
572
+ constructor(pattern) {
573
+ super(pattern);
574
+ }
575
+ static get type() {
576
+ return "exact";
577
+ }
578
+ static get multiRegex() {
579
+ return /^="(.*)"$/;
580
+ }
581
+ static get singleRegex() {
582
+ return /^=(.*)$/;
583
+ }
584
+ search(text) {
585
+ const isMatch = text === this.pattern;
586
+ return {
587
+ isMatch,
588
+ score: isMatch ? 0 : 1,
589
+ indices: [0, this.pattern.length - 1]
590
+ };
591
+ }
592
+ };
593
+ var InverseExactMatch = class extends BaseMatch {
594
+ constructor(pattern) {
595
+ super(pattern);
596
+ }
597
+ static get type() {
598
+ return "inverse-exact";
599
+ }
600
+ static get multiRegex() {
601
+ return /^!"(.*)"$/;
602
+ }
603
+ static get singleRegex() {
604
+ return /^!(.*)$/;
605
+ }
606
+ search(text) {
607
+ const isMatch = text.indexOf(this.pattern) === -1;
608
+ return {
609
+ isMatch,
610
+ score: isMatch ? 0 : 1,
611
+ indices: [0, text.length - 1]
612
+ };
613
+ }
614
+ };
615
+ var PrefixExactMatch = class extends BaseMatch {
616
+ constructor(pattern) {
617
+ super(pattern);
618
+ }
619
+ static get type() {
620
+ return "prefix-exact";
621
+ }
622
+ static get multiRegex() {
623
+ return /^\^"(.*)"$/;
624
+ }
625
+ static get singleRegex() {
626
+ return /^\^(.*)$/;
627
+ }
628
+ search(text) {
629
+ const isMatch = text.startsWith(this.pattern);
630
+ return {
631
+ isMatch,
632
+ score: isMatch ? 0 : 1,
633
+ indices: [0, this.pattern.length - 1]
634
+ };
635
+ }
636
+ };
637
+ var InversePrefixExactMatch = class extends BaseMatch {
638
+ constructor(pattern) {
639
+ super(pattern);
640
+ }
641
+ static get type() {
642
+ return "inverse-prefix-exact";
643
+ }
644
+ static get multiRegex() {
645
+ return /^!\^"(.*)"$/;
646
+ }
647
+ static get singleRegex() {
648
+ return /^!\^(.*)$/;
649
+ }
650
+ search(text) {
651
+ const isMatch = !text.startsWith(this.pattern);
652
+ return {
653
+ isMatch,
654
+ score: isMatch ? 0 : 1,
655
+ indices: [0, text.length - 1]
656
+ };
657
+ }
658
+ };
659
+ var SuffixExactMatch = class extends BaseMatch {
660
+ constructor(pattern) {
661
+ super(pattern);
662
+ }
663
+ static get type() {
664
+ return "suffix-exact";
665
+ }
666
+ static get multiRegex() {
667
+ return /^"(.*)"\$$/;
668
+ }
669
+ static get singleRegex() {
670
+ return /^(.*)\$$/;
671
+ }
672
+ search(text) {
673
+ const isMatch = text.endsWith(this.pattern);
674
+ return {
675
+ isMatch,
676
+ score: isMatch ? 0 : 1,
677
+ indices: [text.length - this.pattern.length, text.length - 1]
678
+ };
679
+ }
680
+ };
681
+ var InverseSuffixExactMatch = class extends BaseMatch {
682
+ constructor(pattern) {
683
+ super(pattern);
684
+ }
685
+ static get type() {
686
+ return "inverse-suffix-exact";
687
+ }
688
+ static get multiRegex() {
689
+ return /^!"(.*)"\$$/;
690
+ }
691
+ static get singleRegex() {
692
+ return /^!(.*)\$$/;
693
+ }
694
+ search(text) {
695
+ const isMatch = !text.endsWith(this.pattern);
696
+ return {
697
+ isMatch,
698
+ score: isMatch ? 0 : 1,
699
+ indices: [0, text.length - 1]
700
+ };
701
+ }
702
+ };
703
+ var FuzzyMatch = class extends BaseMatch {
704
+ constructor(pattern, { location = Config.location, threshold = Config.threshold, distance = Config.distance, includeMatches = Config.includeMatches, findAllMatches = Config.findAllMatches, minMatchCharLength = Config.minMatchCharLength, isCaseSensitive = Config.isCaseSensitive, ignoreDiacritics = Config.ignoreDiacritics, ignoreLocation = Config.ignoreLocation } = {}) {
705
+ super(pattern);
706
+ this._bitapSearch = new BitapSearch(pattern, {
707
+ location,
708
+ threshold,
709
+ distance,
710
+ includeMatches,
711
+ findAllMatches,
712
+ minMatchCharLength,
713
+ isCaseSensitive,
714
+ ignoreDiacritics,
715
+ ignoreLocation
716
+ });
717
+ }
718
+ static get type() {
719
+ return "fuzzy";
720
+ }
721
+ static get multiRegex() {
722
+ return /^"(.*)"$/;
723
+ }
724
+ static get singleRegex() {
725
+ return /^(.*)$/;
726
+ }
727
+ search(text) {
728
+ return this._bitapSearch.searchIn(text);
729
+ }
730
+ };
731
+ var IncludeMatch = class extends BaseMatch {
732
+ constructor(pattern) {
733
+ super(pattern);
734
+ }
735
+ static get type() {
736
+ return "include";
737
+ }
738
+ static get multiRegex() {
739
+ return /^'"(.*)"$/;
740
+ }
741
+ static get singleRegex() {
742
+ return /^'(.*)$/;
743
+ }
744
+ search(text) {
745
+ let location = 0;
746
+ let index;
747
+ const indices = [];
748
+ const patternLen = this.pattern.length;
749
+ while ((index = text.indexOf(this.pattern, location)) > -1) {
750
+ location = index + patternLen;
751
+ indices.push([index, location - 1]);
752
+ }
753
+ const isMatch = !!indices.length;
754
+ return {
755
+ isMatch,
756
+ score: isMatch ? 0 : 1,
757
+ indices
758
+ };
759
+ }
760
+ };
761
+ const searchers = [
762
+ ExactMatch,
763
+ IncludeMatch,
764
+ PrefixExactMatch,
765
+ InversePrefixExactMatch,
766
+ InverseSuffixExactMatch,
767
+ SuffixExactMatch,
768
+ InverseExactMatch,
769
+ FuzzyMatch
770
+ ];
771
+ const searchersLen = searchers.length;
772
+ const SPACE_RE = / +(?=(?:[^\"]*\"[^\"]*\")*[^\"]*$)/;
773
+ const OR_TOKEN = "|";
774
+ function parseQuery(pattern, options = {}) {
775
+ return pattern.split(OR_TOKEN).map((item) => {
776
+ let query = item.trim().split(SPACE_RE).filter((item$1) => item$1 && !!item$1.trim());
777
+ let results = [];
778
+ for (let i = 0, len = query.length; i < len; i += 1) {
779
+ const queryItem = query[i];
780
+ let found = false;
781
+ let idx = -1;
782
+ while (!found && ++idx < searchersLen) {
783
+ const searcher = searchers[idx];
784
+ let token = searcher.isMultiMatch(queryItem);
785
+ if (token) {
786
+ results.push(new searcher(token, options));
787
+ found = true;
788
+ }
789
+ }
790
+ if (found) continue;
791
+ idx = -1;
792
+ while (++idx < searchersLen) {
793
+ const searcher = searchers[idx];
794
+ let token = searcher.isSingleMatch(queryItem);
795
+ if (token) {
796
+ results.push(new searcher(token, options));
797
+ break;
798
+ }
799
+ }
800
+ }
801
+ return results;
802
+ });
803
+ }
804
+ const MultiMatchSet = new Set([FuzzyMatch.type, IncludeMatch.type]);
805
+ /**
806
+ * Command-like searching
807
+ * ======================
808
+ *
809
+ * Given multiple search terms delimited by spaces.e.g. `^jscript .python$ ruby !java`,
810
+ * search in a given text.
811
+ *
812
+ * Search syntax:
813
+ *
814
+ * | Token | Match type | Description |
815
+ * | ----------- | -------------------------- | -------------------------------------- |
816
+ * | `jscript` | fuzzy-match | Items that fuzzy match `jscript` |
817
+ * | `=scheme` | exact-match | Items that are `scheme` |
818
+ * | `'python` | include-match | Items that include `python` |
819
+ * | `!ruby` | inverse-exact-match | Items that do not include `ruby` |
820
+ * | `^java` | prefix-exact-match | Items that start with `java` |
821
+ * | `!^earlang` | inverse-prefix-exact-match | Items that do not start with `earlang` |
822
+ * | `.js$` | suffix-exact-match | Items that end with `.js` |
823
+ * | `!.go$` | inverse-suffix-exact-match | Items that do not end with `.go` |
824
+ *
825
+ * A single pipe character acts as an OR operator. For example, the following
826
+ * query matches entries that start with `core` and end with either`go`, `rb`,
827
+ * or`py`.
828
+ *
829
+ * ```
830
+ * ^core go$ | rb$ | py$
831
+ * ```
832
+ */
833
+ var ExtendedSearch = class {
834
+ constructor(pattern, { isCaseSensitive = Config.isCaseSensitive, ignoreDiacritics = Config.ignoreDiacritics, includeMatches = Config.includeMatches, minMatchCharLength = Config.minMatchCharLength, ignoreLocation = Config.ignoreLocation, findAllMatches = Config.findAllMatches, location = Config.location, threshold = Config.threshold, distance = Config.distance } = {}) {
835
+ this.query = null;
836
+ this.options = {
837
+ isCaseSensitive,
838
+ ignoreDiacritics,
839
+ includeMatches,
840
+ minMatchCharLength,
841
+ findAllMatches,
842
+ ignoreLocation,
843
+ location,
844
+ threshold,
845
+ distance
846
+ };
847
+ pattern = isCaseSensitive ? pattern : pattern.toLowerCase();
848
+ pattern = ignoreDiacritics ? stripDiacritics(pattern) : pattern;
849
+ this.pattern = pattern;
850
+ this.query = parseQuery(this.pattern, this.options);
851
+ }
852
+ static condition(_, options) {
853
+ return options.useExtendedSearch;
854
+ }
855
+ searchIn(text) {
856
+ const query = this.query;
857
+ if (!query) return {
858
+ isMatch: false,
859
+ score: 1
860
+ };
861
+ const { includeMatches, isCaseSensitive, ignoreDiacritics } = this.options;
862
+ text = isCaseSensitive ? text : text.toLowerCase();
863
+ text = ignoreDiacritics ? stripDiacritics(text) : text;
864
+ let numMatches = 0;
865
+ let allIndices = [];
866
+ let totalScore = 0;
867
+ for (let i = 0, qLen = query.length; i < qLen; i += 1) {
868
+ const searchers$1 = query[i];
869
+ allIndices.length = 0;
870
+ numMatches = 0;
871
+ for (let j = 0, pLen = searchers$1.length; j < pLen; j += 1) {
872
+ const searcher = searchers$1[j];
873
+ const { isMatch, indices, score } = searcher.search(text);
874
+ if (isMatch) {
875
+ numMatches += 1;
876
+ totalScore += score;
877
+ if (includeMatches) {
878
+ const type = searcher.constructor.type;
879
+ if (MultiMatchSet.has(type)) allIndices = [...allIndices, ...indices];
880
+ else allIndices.push(indices);
881
+ }
882
+ } else {
883
+ totalScore = 0;
884
+ numMatches = 0;
885
+ allIndices.length = 0;
886
+ break;
887
+ }
888
+ }
889
+ if (numMatches) {
890
+ let result = {
891
+ isMatch: true,
892
+ score: totalScore / numMatches
893
+ };
894
+ if (includeMatches) result.indices = allIndices;
895
+ return result;
896
+ }
897
+ }
898
+ return {
899
+ isMatch: false,
900
+ score: 1
901
+ };
902
+ }
903
+ };
904
+ const registeredSearchers = [];
905
+ function register(...args) {
906
+ registeredSearchers.push(...args);
907
+ }
908
+ function createSearcher(pattern, options) {
909
+ for (let i = 0, len = registeredSearchers.length; i < len; i += 1) {
910
+ let searcherClass = registeredSearchers[i];
911
+ if (searcherClass.condition(pattern, options)) return new searcherClass(pattern, options);
912
+ }
913
+ return new BitapSearch(pattern, options);
914
+ }
915
+ const LogicalOperator = {
916
+ AND: "$and",
917
+ OR: "$or"
918
+ };
919
+ const KeyType = {
920
+ PATH: "$path",
921
+ PATTERN: "$val"
922
+ };
923
+ const isExpression = (query) => !!(query[LogicalOperator.AND] || query[LogicalOperator.OR]);
924
+ const isPath = (query) => !!query[KeyType.PATH];
925
+ const isLeaf = (query) => !isArray(query) && isObject(query) && !isExpression(query);
926
+ const convertToExplicit = (query) => ({ [LogicalOperator.AND]: Object.keys(query).map((key) => ({ [key]: query[key] })) });
927
+ function parse(query, options, { auto = true } = {}) {
928
+ const next = (query$1) => {
929
+ let keys = Object.keys(query$1);
930
+ const isQueryPath = isPath(query$1);
931
+ if (!isQueryPath && keys.length > 1 && !isExpression(query$1)) return next(convertToExplicit(query$1));
932
+ if (isLeaf(query$1)) {
933
+ const key = isQueryPath ? query$1[KeyType.PATH] : keys[0];
934
+ const pattern = isQueryPath ? query$1[KeyType.PATTERN] : query$1[key];
935
+ if (!isString(pattern)) throw new Error(LOGICAL_SEARCH_INVALID_QUERY_FOR_KEY(key));
936
+ const obj = {
937
+ keyId: createKeyId(key),
938
+ pattern
939
+ };
940
+ if (auto) obj.searcher = createSearcher(pattern, options);
941
+ return obj;
942
+ }
943
+ let node = {
944
+ children: [],
945
+ operator: keys[0]
946
+ };
947
+ keys.forEach((key) => {
948
+ const value = query$1[key];
949
+ if (isArray(value)) value.forEach((item) => {
950
+ node.children.push(next(item));
951
+ });
952
+ });
953
+ return node;
954
+ };
955
+ if (!isExpression(query)) query = convertToExplicit(query);
956
+ return next(query);
957
+ }
958
+ function computeScore(results, { ignoreFieldNorm = Config.ignoreFieldNorm }) {
959
+ results.forEach((result) => {
960
+ let totalScore = 1;
961
+ result.matches.forEach(({ key, norm: norm$1, score }) => {
962
+ const weight = key ? key.weight : null;
963
+ totalScore *= Math.pow(score === 0 && weight ? Number.EPSILON : score, (weight || 1) * (ignoreFieldNorm ? 1 : norm$1));
964
+ });
965
+ result.score = totalScore;
966
+ });
967
+ }
968
+ function transformMatches(result, data) {
969
+ const matches = result.matches;
970
+ data.matches = [];
971
+ if (!isDefined(matches)) return;
972
+ matches.forEach((match) => {
973
+ if (!isDefined(match.indices) || !match.indices.length) return;
974
+ const { indices, value } = match;
975
+ let obj = {
976
+ indices,
977
+ value
978
+ };
979
+ if (match.key) obj.key = match.key.src;
980
+ if (match.idx > -1) obj.refIndex = match.idx;
981
+ data.matches.push(obj);
982
+ });
983
+ }
984
+ function transformScore(result, data) {
985
+ data.score = result.score;
986
+ }
987
+ function format(results, docs, { includeMatches = Config.includeMatches, includeScore = Config.includeScore } = {}) {
988
+ const transformers = [];
989
+ if (includeMatches) transformers.push(transformMatches);
990
+ if (includeScore) transformers.push(transformScore);
991
+ return results.map((result) => {
992
+ const { idx } = result;
993
+ const data = {
994
+ item: docs[idx],
995
+ refIndex: idx
996
+ };
997
+ if (transformers.length) transformers.forEach((transformer) => {
998
+ transformer(result, data);
999
+ });
1000
+ return data;
1001
+ });
1002
+ }
1003
+ var Fuse = class {
1004
+ constructor(docs, options = {}, index) {
1005
+ this.options = {
1006
+ ...Config,
1007
+ ...options
1008
+ };
1009
+ if (this.options.useExtendedSearch && false);
1010
+ this._keyStore = new KeyStore(this.options.keys);
1011
+ this.setCollection(docs, index);
1012
+ }
1013
+ setCollection(docs, index) {
1014
+ this._docs = docs;
1015
+ if (index && !(index instanceof FuseIndex)) throw new Error(INCORRECT_INDEX_TYPE);
1016
+ this._myIndex = index || createIndex(this.options.keys, this._docs, {
1017
+ getFn: this.options.getFn,
1018
+ fieldNormWeight: this.options.fieldNormWeight
1019
+ });
1020
+ }
1021
+ add(doc) {
1022
+ if (!isDefined(doc)) return;
1023
+ this._docs.push(doc);
1024
+ this._myIndex.add(doc);
1025
+ }
1026
+ remove(predicate = () => false) {
1027
+ const results = [];
1028
+ for (let i = 0, len = this._docs.length; i < len; i += 1) {
1029
+ const doc = this._docs[i];
1030
+ if (predicate(doc, i)) {
1031
+ this.removeAt(i);
1032
+ i -= 1;
1033
+ len -= 1;
1034
+ results.push(doc);
1035
+ }
1036
+ }
1037
+ return results;
1038
+ }
1039
+ removeAt(idx) {
1040
+ this._docs.splice(idx, 1);
1041
+ this._myIndex.removeAt(idx);
1042
+ }
1043
+ getIndex() {
1044
+ return this._myIndex;
1045
+ }
1046
+ search(query, { limit = -1 } = {}) {
1047
+ const { includeMatches, includeScore, shouldSort, sortFn, ignoreFieldNorm } = this.options;
1048
+ let results = isString(query) ? isString(this._docs[0]) ? this._searchStringList(query) : this._searchObjectList(query) : this._searchLogical(query);
1049
+ computeScore(results, { ignoreFieldNorm });
1050
+ if (shouldSort) results.sort(sortFn);
1051
+ if (isNumber(limit) && limit > -1) results = results.slice(0, limit);
1052
+ return format(results, this._docs, {
1053
+ includeMatches,
1054
+ includeScore
1055
+ });
1056
+ }
1057
+ _searchStringList(query) {
1058
+ const searcher = createSearcher(query, this.options);
1059
+ const { records } = this._myIndex;
1060
+ const results = [];
1061
+ records.forEach(({ v: text, i: idx, n: norm$1 }) => {
1062
+ if (!isDefined(text)) return;
1063
+ const { isMatch, score, indices } = searcher.searchIn(text);
1064
+ if (isMatch) results.push({
1065
+ item: text,
1066
+ idx,
1067
+ matches: [{
1068
+ score,
1069
+ value: text,
1070
+ norm: norm$1,
1071
+ indices
1072
+ }]
1073
+ });
1074
+ });
1075
+ return results;
1076
+ }
1077
+ _searchLogical(query) {
1078
+ const expression = parse(query, this.options);
1079
+ const evaluate = (node, item, idx) => {
1080
+ if (!node.children) {
1081
+ const { keyId, searcher } = node;
1082
+ const matches = this._findMatches({
1083
+ key: this._keyStore.get(keyId),
1084
+ value: this._myIndex.getValueForItemAtKeyId(item, keyId),
1085
+ searcher
1086
+ });
1087
+ if (matches && matches.length) return [{
1088
+ idx,
1089
+ item,
1090
+ matches
1091
+ }];
1092
+ return [];
1093
+ }
1094
+ const res = [];
1095
+ for (let i = 0, len = node.children.length; i < len; i += 1) {
1096
+ const child = node.children[i];
1097
+ const result = evaluate(child, item, idx);
1098
+ if (result.length) res.push(...result);
1099
+ else if (node.operator === LogicalOperator.AND) return [];
1100
+ }
1101
+ return res;
1102
+ };
1103
+ const records = this._myIndex.records;
1104
+ const resultMap = {};
1105
+ const results = [];
1106
+ records.forEach(({ $: item, i: idx }) => {
1107
+ if (isDefined(item)) {
1108
+ let expResults = evaluate(expression, item, idx);
1109
+ if (expResults.length) {
1110
+ if (!resultMap[idx]) {
1111
+ resultMap[idx] = {
1112
+ idx,
1113
+ item,
1114
+ matches: []
1115
+ };
1116
+ results.push(resultMap[idx]);
1117
+ }
1118
+ expResults.forEach(({ matches }) => {
1119
+ resultMap[idx].matches.push(...matches);
1120
+ });
1121
+ }
1122
+ }
1123
+ });
1124
+ return results;
1125
+ }
1126
+ _searchObjectList(query) {
1127
+ const searcher = createSearcher(query, this.options);
1128
+ const { keys, records } = this._myIndex;
1129
+ const results = [];
1130
+ records.forEach(({ $: item, i: idx }) => {
1131
+ if (!isDefined(item)) return;
1132
+ let matches = [];
1133
+ keys.forEach((key, keyIndex) => {
1134
+ matches.push(...this._findMatches({
1135
+ key,
1136
+ value: item[keyIndex],
1137
+ searcher
1138
+ }));
1139
+ });
1140
+ if (matches.length) results.push({
1141
+ idx,
1142
+ item,
1143
+ matches
1144
+ });
1145
+ });
1146
+ return results;
1147
+ }
1148
+ _findMatches({ key, value, searcher }) {
1149
+ if (!isDefined(value)) return [];
1150
+ let matches = [];
1151
+ if (isArray(value)) value.forEach(({ v: text, i: idx, n: norm$1 }) => {
1152
+ if (!isDefined(text)) return;
1153
+ const { isMatch, score, indices } = searcher.searchIn(text);
1154
+ if (isMatch) matches.push({
1155
+ score,
1156
+ key,
1157
+ value: text,
1158
+ idx,
1159
+ norm: norm$1,
1160
+ indices
1161
+ });
1162
+ });
1163
+ else {
1164
+ const { v: text, n: norm$1 } = value;
1165
+ const { isMatch, score, indices } = searcher.searchIn(text);
1166
+ if (isMatch) matches.push({
1167
+ score,
1168
+ key,
1169
+ value: text,
1170
+ norm: norm$1,
1171
+ indices
1172
+ });
1173
+ }
1174
+ return matches;
1175
+ }
1176
+ };
1177
+ Fuse.version = "7.1.0";
1178
+ Fuse.createIndex = createIndex;
1179
+ Fuse.parseIndex = parseIndex;
1180
+ Fuse.config = Config;
1181
+ Fuse.parseQuery = parse;
1182
+ register(ExtendedSearch);
1183
+
1184
+ //#endregion
1185
+ //#region src/hooks/Search/useSearchInput.ts
1186
+ const useSearchInput = ({ initialSearchText = "", searchOrigin, onSearchSubmit }) => {
1187
+ const track = (0, jotai.useAtomValue)(require_chat.amplitudeTrackEventAtom);
1188
+ const [searchText, setSearchText] = (0, react.useState)(initialSearchText);
1189
+ const [hasText, setHasText] = (0, react.useState)(initialSearchText.trim().length > 0);
1190
+ const [isFocused, setIsFocused] = (0, react.useState)(false);
1191
+ const [focusedIndex, setFocusedIndex] = (0, react.useState)(-1);
1192
+ const [focusedOptionId, setFocusedOptionId] = (0, react.useState)(void 0);
1193
+ const globalAutocompleteResults = (0, jotai.useAtomValue)(require_globalSearch.autocompleteStateAtom).results;
1194
+ const isAutocompleteEnabled = globalAutocompleteResults.length > 0;
1195
+ const isSelectingAutocomplete = (0, react.useRef)(false);
1196
+ const shouldShowAutocomplete = hasText && isFocused && isAutocompleteEnabled;
1197
+ const autocompleteResultsLimit = 5;
1198
+ const fuseOptions = (0, react.useMemo)(() => ({
1199
+ includeScore: true,
1200
+ includeMatches: true,
1201
+ threshold: .3,
1202
+ location: 0,
1203
+ distance: 4,
1204
+ minMatchCharLength: 1,
1205
+ keys: [],
1206
+ useExtendedSearch: false,
1207
+ ignoreLocation: false
1208
+ }), []);
1209
+ const fuse = (0, react.useMemo)(() => {
1210
+ if (!globalAutocompleteResults.length) return null;
1211
+ return new Fuse(globalAutocompleteResults, fuseOptions);
1212
+ }, [globalAutocompleteResults, fuseOptions]);
1213
+ const autocompleteResults = (0, react.useMemo)(() => {
1214
+ if (!isAutocompleteEnabled || !fuse) return [];
1215
+ if (!searchText.trim()) return [];
1216
+ return fuse.search(searchText).slice(0, autocompleteResultsLimit).map((result) => result.item);
1217
+ }, [
1218
+ isAutocompleteEnabled,
1219
+ fuse,
1220
+ searchText,
1221
+ autocompleteResultsLimit
1222
+ ]);
1223
+ const handleSearchInputChange = (0, react.useCallback)((newValue) => {
1224
+ if (newValue !== searchText) setIsFocused(true);
1225
+ if (newValue.length === 1) track?.({
1226
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchInputStarted,
1227
+ eventProps: { searchOrigin }
1228
+ });
1229
+ setSearchText(newValue);
1230
+ setHasText(newValue.trim().length > 0);
1231
+ setFocusedIndex(-1);
1232
+ setFocusedOptionId(void 0);
1233
+ }, [
1234
+ searchOrigin,
1235
+ searchText,
1236
+ track
1237
+ ]);
1238
+ const handleSearchInputFocus = (0, react.useCallback)(() => {
1239
+ setIsFocused(true);
1240
+ }, []);
1241
+ const handleSearchInputBlur = (0, react.useCallback)(() => {
1242
+ setTimeout(() => {
1243
+ if (!isSelectingAutocomplete.current) {
1244
+ setIsFocused(false);
1245
+ setFocusedIndex(-1);
1246
+ setFocusedOptionId(void 0);
1247
+ }
1248
+ }, 150);
1249
+ }, []);
1250
+ const handleSubmitSearch = (0, react.useCallback)(() => {
1251
+ setIsFocused(false);
1252
+ if (searchText.trim() && onSearchSubmit) {
1253
+ track?.({
1254
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchQuerySubmitted,
1255
+ eventProps: {
1256
+ searchOrigin,
1257
+ queryText: searchText.trim()
1258
+ },
1259
+ alsoSendToGoogleAnalytics: true
1260
+ });
1261
+ onSearchSubmit(searchText.trim());
1262
+ }
1263
+ }, [
1264
+ searchText,
1265
+ onSearchSubmit,
1266
+ searchOrigin,
1267
+ track
1268
+ ]);
1269
+ const handleAutocompleteSelect = (0, react.useCallback)((suggestion, rankPosition) => {
1270
+ isSelectingAutocomplete.current = true;
1271
+ setSearchText(suggestion);
1272
+ setHasText(suggestion.trim().length > 0);
1273
+ setIsFocused(false);
1274
+ track?.({
1275
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchInputStarted,
1276
+ eventProps: { searchOrigin }
1277
+ });
1278
+ if (onSearchSubmit) onSearchSubmit(suggestion.trim());
1279
+ setTimeout(() => {
1280
+ isSelectingAutocomplete.current = false;
1281
+ }, 100);
1282
+ }, [
1283
+ onSearchSubmit,
1284
+ searchOrigin,
1285
+ track
1286
+ ]);
1287
+ const handleKeyDown = (0, react.useCallback)((event) => {
1288
+ if (event.key === "ArrowDown") {
1289
+ event.preventDefault();
1290
+ const newIndex = (focusedIndex + 1) % autocompleteResults.length;
1291
+ setFocusedIndex(newIndex);
1292
+ setFocusedOptionId(`option-${newIndex}`);
1293
+ } else if (event.key === "ArrowUp") {
1294
+ event.preventDefault();
1295
+ const newIndex = (focusedIndex - 1 + autocompleteResults.length) % autocompleteResults.length;
1296
+ setFocusedIndex(newIndex);
1297
+ setFocusedOptionId(`option-${newIndex}`);
1298
+ } else if (event.key === "Enter") if (focusedIndex === -1) {
1299
+ event.preventDefault();
1300
+ handleSubmitSearch();
1301
+ } else {
1302
+ event.preventDefault();
1303
+ const suggestionText = autocompleteResults[focusedIndex];
1304
+ handleAutocompleteSelect(suggestionText, focusedIndex);
1305
+ }
1306
+ else if (event.key === "Escape") {
1307
+ event.preventDefault();
1308
+ setFocusedIndex(-1);
1309
+ setFocusedOptionId(void 0);
1310
+ }
1311
+ }, [
1312
+ autocompleteResults,
1313
+ focusedIndex,
1314
+ handleAutocompleteSelect,
1315
+ handleSubmitSearch
1316
+ ]);
1317
+ const setSearchTextUtility = (0, react.useCallback)((text) => {
1318
+ setSearchText(text);
1319
+ setHasText(text.trim().length > 0);
1320
+ }, []);
1321
+ const resetSearch = (0, react.useCallback)(() => {
1322
+ setSearchText("");
1323
+ setHasText(false);
1324
+ setIsFocused(false);
1325
+ setFocusedIndex(-1);
1326
+ setFocusedOptionId(void 0);
1327
+ }, []);
1328
+ return {
1329
+ searchText,
1330
+ hasText,
1331
+ isFocused,
1332
+ focusedIndex,
1333
+ focusedOptionId,
1334
+ autocompleteResults,
1335
+ shouldShowAutocomplete,
1336
+ handleSearchInputChange,
1337
+ handleSearchInputFocus,
1338
+ handleSearchInputBlur,
1339
+ handleKeyDown,
1340
+ handleAutocompleteSelect,
1341
+ handleSubmitSearch,
1342
+ setSearchText: setSearchTextUtility,
1343
+ resetSearch
1344
+ };
1345
+ };
1346
+
1347
+ //#endregion
1348
+ //#region src/atoms/search/productRetrievalAdapter.ts
1349
+ let productRetrievalFunction = null;
1350
+ const getProductRetrievalFunction = () => {
1351
+ if (!productRetrievalFunction) throw new Error("Product retrieval function not initialized. Make sure ProductRetrievalProvider is mounted.");
1352
+ return productRetrievalFunction;
1353
+ };
1354
+
1355
+ //#endregion
1356
+ //#region src/atoms/search/productRetrievalAPI.ts
1357
+ const productRetrievalAtom = (0, jotai.atom)({
1358
+ data: null,
1359
+ loading: false,
1360
+ error: null,
1361
+ lastProductIds: null
1362
+ });
1363
+ const performProductRetrievalAtom = (0, jotai.atom)(null, async (get$1, set, params) => {
1364
+ if (get$1(productRetrievalAtom).loading) return;
1365
+ set(productRetrievalAtom, {
1366
+ data: null,
1367
+ loading: true,
1368
+ error: null,
1369
+ lastProductIds: params.productIds
1370
+ });
1371
+ try {
1372
+ const result = await getProductRetrievalFunction()(params);
1373
+ set(productRetrievalAtom, {
1374
+ data: result,
1375
+ loading: false,
1376
+ error: null,
1377
+ lastProductIds: params.productIds
1378
+ });
1379
+ } catch (error) {
1380
+ const errorMessage = error instanceof Error ? error.message : "An unknown error occurred";
1381
+ set(productRetrievalAtom, {
1382
+ data: null,
1383
+ loading: false,
1384
+ error: errorMessage,
1385
+ lastProductIds: params.productIds
1386
+ });
1387
+ }
1388
+ });
1389
+ const retrievedProductsAtom = (0, jotai.atom)((get$1) => {
1390
+ return get$1(productRetrievalAtom).data?.products || [];
1391
+ });
1392
+ const productRetrievalLoadingAtom = (0, jotai.atom)((get$1) => {
1393
+ return get$1(productRetrievalAtom).loading;
1394
+ });
1395
+ const productRetrievalErrorAtom = (0, jotai.atom)((get$1) => {
1396
+ return get$1(productRetrievalAtom).error;
1397
+ });
1398
+
1399
+ //#endregion
1400
+ //#region src/hooks/Search/useRecommendedProducts.ts
1401
+ /**
1402
+ * A hook to get recommended products.
1403
+ * It checks the `IsRecommendedProductsEnabled` feature gate and triggers fetching if needed.
1404
+ * If the gate is enabled, it returns the list of recommended products.
1405
+ * If the gate is disabled, it returns an empty array.
1406
+ *
1407
+ * @returns The list of recommended products or an empty array.
1408
+ */
1409
+ const useRecommendedProducts = () => {
1410
+ const { featureFlagService } = require_featureFlagServiceContext.useFeatureFlagService();
1411
+ const isEnabled = featureFlagService?.isFeatureGateEnabled(require_featureGates.FeatureGates.IsRecommendedProductsEnabled);
1412
+ const recommendedProducts = (0, jotai.useAtomValue)(retrievedProductsAtom);
1413
+ const performProductRetrieval = (0, jotai.useSetAtom)(performProductRetrievalAtom);
1414
+ const { lastProductIds } = (0, jotai.useAtomValue)(productRetrievalAtom);
1415
+ const orgUIConfig = require_NewOrgConfig.useNewOrgConfig();
1416
+ (0, react.useEffect)(() => {
1417
+ const recommendedProductIds = orgUIConfig.frontendConfig?.uiConfigs?.searchConfig ? orgUIConfig.frontendConfig?.uiConfigs?.searchConfig.recommendedProducts : [];
1418
+ if (isEnabled && recommendedProductIds && recommendedProductIds.length > 0 && !lastProductIds) performProductRetrieval({ productIds: recommendedProductIds });
1419
+ }, [
1420
+ isEnabled,
1421
+ performProductRetrieval,
1422
+ lastProductIds,
1423
+ orgUIConfig
1424
+ ]);
1425
+ return isEnabled ? recommendedProducts : [];
1426
+ };
1427
+
1428
+ //#endregion
45
1429
  //#region src/hooks/Search/useSearch.tsx
46
1430
  const useSearch = () => {
47
1431
  const config = require_NewOrgConfig.useNewOrgConfig();
@@ -49,7 +1433,6 @@ const useSearch = () => {
49
1433
  const { data: searchData, loading: isLoadingSearch } = (0, jotai.useAtomValue)(require_search.searchAtom);
50
1434
  const productList = (0, jotai.useAtomValue)(require_search.filteredSearchProductsAtom);
51
1435
  const performSearch = (0, jotai.useSetAtom)(require_search.performSearchAtom);
52
- const [{ results: autocompleteResults, isLoading: isLoadingAutocomplete }, setAutocompleteState] = (0, jotai.useAtom)(require_globalSearch.autocompleteStateAtom);
53
1436
  const [{ query }] = (0, jotai.useAtom)(require_search.searchParamsAtom);
54
1437
  const [isFilterOpen, setIsFilterOpen] = (0, jotai.useAtom)(require_globalSearch.isFilterOpenAtom);
55
1438
  const [selectedFilterOptions] = (0, jotai.useAtom)(require_search.searchSelectedFiltersAtom);
@@ -58,12 +1441,26 @@ const useSearch = () => {
58
1441
  const [productSorting, setProductSorting] = (0, jotai.useAtom)(require_search.searchProductSortingAtom);
59
1442
  const clearFilters = (0, jotai.useSetAtom)(require_search.clearSearchFiltersAtom);
60
1443
  const searchFilters = (0, jotai.useAtomValue)(require_search.searchFiltersAtom);
61
- const [isDirty, setIsDirty] = (0, react.useState)(true);
62
- const [focusedIndex, setFocusedIndex] = (0, react.useState)(-1);
63
- const [focusedOptionId, setFocusedOptionId] = (0, react.useState)(void 0);
64
- const [searchText, setSearchText] = (0, react.useState)(query || "");
1444
+ const recommendedProducts = useRecommendedProducts();
65
1445
  const searchResultsRef = (0, react.useRef)(null);
66
- const debouncedSearchText = require_useDebounce.useDebounce(searchText, 200);
1446
+ const scrollToTop = (0, react.useCallback)(() => {
1447
+ const container = searchResultsRef.current;
1448
+ if (container) container.scrollTo({
1449
+ top: 0,
1450
+ behavior: "smooth"
1451
+ });
1452
+ else window.scrollTo({
1453
+ top: 0,
1454
+ behavior: "smooth"
1455
+ });
1456
+ }, []);
1457
+ const { searchText, focusedIndex, focusedOptionId, autocompleteResults, shouldShowAutocomplete, handleSearchInputChange, handleSearchInputFocus, handleSearchInputBlur, handleKeyDown, handleAutocompleteSelect, handleSubmitSearch, setSearchText } = useSearchInput({
1458
+ initialSearchText: query || "",
1459
+ searchOrigin: require_models.SpiffyWidgets.SearchResults,
1460
+ onSearchSubmit: (searchQuery) => {
1461
+ performSearch({ query: searchQuery });
1462
+ }
1463
+ });
67
1464
  const searchResultsState = require_utils.getSearchResultsState(isLoadingSearch, searchData);
68
1465
  const dynamicFilters = searchData?.filters || [];
69
1466
  const safeProductCardConfig = config?.frontendConfig?.uiConfigs?.productCardConfig || {
@@ -123,76 +1520,16 @@ const useSearch = () => {
123
1520
  }
124
1521
  });
125
1522
  addFilter(require_search.createFilterOption("dynamic", filter, dynamicFilterDisplayName));
1523
+ scrollToTop();
126
1524
  }, [
127
1525
  addFilter,
128
1526
  searchText,
129
- trackEvent
1527
+ scrollToTop
130
1528
  ]);
131
1529
  const handleRemoveFilter = (0, react.useCallback)((filter) => {
132
1530
  removeFilter(filter.id);
133
- }, [removeFilter]);
134
- const handleSubmitSearch = (0, react.useCallback)(async () => {
135
- if (searchText.trim()) {
136
- trackEvent({
137
- eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchQuerySubmitted,
138
- eventProps: {
139
- searchOrigin: require_models.SpiffyWidgets.SearchResults,
140
- queryText: searchText.trim()
141
- },
142
- alsoSendToGoogleAnalytics: true
143
- });
144
- const url = new URL(window.location.href);
145
- url.searchParams.set("esq", searchText.trim());
146
- window.history.pushState({}, "", url);
147
- performSearch({ query: searchText.trim() });
148
- }
149
- }, [
150
- performSearch,
151
- searchText,
152
- trackEvent
153
- ]);
154
- const handleAutocompleteSelect = (0, react.useCallback)((suggestion) => {
155
- setSearchText(suggestion);
156
- handleSubmitSearch();
157
- }, [handleSubmitSearch, setSearchText]);
158
- const handleKeyDown = (0, react.useCallback)((event) => {
159
- if (event.key === "ArrowDown") {
160
- event.preventDefault();
161
- const newIndex = (focusedIndex + 1) % autocompleteResults.length;
162
- setFocusedIndex(newIndex);
163
- setFocusedOptionId(`option-${newIndex}`);
164
- } else if (event.key === "ArrowUp") {
165
- event.preventDefault();
166
- const newIndex = (focusedIndex - 1 + autocompleteResults.length) % autocompleteResults.length;
167
- setFocusedIndex(newIndex);
168
- setFocusedOptionId(`option-${newIndex}`);
169
- } else if (event.key === "Enter") if (focusedIndex === -1) {
170
- event.preventDefault();
171
- handleSubmitSearch();
172
- } else {
173
- event.preventDefault();
174
- const suggestionText = autocompleteResults[focusedIndex];
175
- handleAutocompleteSelect(suggestionText);
176
- }
177
- else if (event.key === "Escape") {
178
- event.preventDefault();
179
- setFocusedIndex(-1);
180
- setFocusedOptionId(void 0);
181
- }
182
- }, [
183
- autocompleteResults,
184
- focusedIndex,
185
- handleAutocompleteSelect,
186
- handleSubmitSearch
187
- ]);
188
- const handleSearchInputChange = (newValue) => {
189
- if (newValue.length === 1) trackEvent({
190
- eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchInputStarted,
191
- eventProps: { searchOrigin: require_models.SpiffyWidgets.SearchResults }
192
- });
193
- setSearchText(newValue);
194
- setIsDirty(true);
195
- };
1531
+ scrollToTop();
1532
+ }, [removeFilter, scrollToTop]);
196
1533
  const handleSelectFilterItem = (0, react.useCallback)(({ filterId, filterItemId, isSelected, displayName }) => {
197
1534
  if (filterId === "sort") {
198
1535
  const newSort = filterItemId;
@@ -204,8 +1541,11 @@ const useSearch = () => {
204
1541
  }
205
1542
  });
206
1543
  setProductSorting(newSort);
207
- } else if (!isSelected) removeFilter(`${filterId}:${filterItemId}`);
208
- else {
1544
+ scrollToTop();
1545
+ } else if (!isSelected) {
1546
+ removeFilter(`${filterId}:${filterItemId}`);
1547
+ scrollToTop();
1548
+ } else {
209
1549
  trackEvent({
210
1550
  eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchFilterClicked,
211
1551
  eventProps: {
@@ -216,78 +1556,44 @@ const useSearch = () => {
216
1556
  }
217
1557
  });
218
1558
  addFilter(require_search.createFilterOption(filterId, filterItemId, displayName));
1559
+ scrollToTop();
219
1560
  }
220
1561
  }, [
221
1562
  addFilter,
222
1563
  removeFilter,
223
1564
  setProductSorting,
224
1565
  searchText,
225
- trackEvent
1566
+ scrollToTop
226
1567
  ]);
227
1568
  const handleClearAllFilters = (0, react.useCallback)(() => {
228
1569
  setProductSorting(require_types$1.ProductSorting.FEATURED);
229
1570
  clearFilters();
230
- }, [setProductSorting, clearFilters]);
1571
+ scrollToTop();
1572
+ }, [
1573
+ setProductSorting,
1574
+ clearFilters,
1575
+ scrollToTop
1576
+ ]);
231
1577
  require_TrackComponentVisibleEvent.useTrackComponentVisibleEvent(require_models.SpiffyWidgets.SearchResults, searchResultsRef, {}, require_amplitudeContext.SpiffyMetricsEventName.SearchComponentVisible);
232
1578
  (0, react.useEffect)(() => {
233
- if (searchResultsState === require_utils.SearchResultsState.NoResults || searchResultsState === require_utils.SearchResultsState.Results) trackEvent({
1579
+ if (searchResultsState === require_utils.SearchResultsState.Results || searchResultsState === require_utils.SearchResultsState.NoResults) trackEvent({
234
1580
  eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchResultsViewed,
235
1581
  eventProps: {
236
1582
  queryText: searchText,
237
1583
  resultsCount: productList.length
238
1584
  }
239
1585
  });
240
- }, [
241
- productList.length,
242
- searchResultsState,
243
- trackEvent
244
- ]);
245
- (0, react.useEffect)(() => {
246
- if (query && query !== searchText) setSearchText(query);
247
- }, [query]);
248
- (0, react.useEffect)(() => {
249
- const esq = new URLSearchParams(window.location.search).get("esq");
250
- if (esq) {
251
- setSearchText(esq);
252
- performSearch({ query: esq });
253
- }
254
- }, [performSearch]);
255
- const fetchAutocompleteSuggestions = (_query) => {
256
- return Promise.resolve([]);
257
- };
1586
+ }, [productList.length, searchResultsState]);
258
1587
  (0, react.useEffect)(() => {
259
- if (fetchAutocompleteSuggestions === void 0) return;
260
- if (!isDirty || debouncedSearchText.length <= 2) {
261
- setAutocompleteState({
262
- results: [],
263
- isLoading: false
264
- });
265
- return;
1588
+ if (query && query !== searchText) {
1589
+ const esq = new URLSearchParams(window.location.search).get("esq");
1590
+ setSearchText(query);
1591
+ performSearch({ query: esq ?? query });
266
1592
  }
267
- setAutocompleteState((prev) => ({
268
- ...prev,
269
- isLoading: true
270
- }));
271
- const fetchData = async () => {
272
- try {
273
- const results = await fetchAutocompleteSuggestions?.(debouncedSearchText);
274
- setAutocompleteState({
275
- results: results ?? [],
276
- isLoading: false
277
- });
278
- } catch (error) {
279
- require_logger.logger_default.logError("Failed to fetch autocomplete suggestions:", error);
280
- setAutocompleteState({
281
- results: [],
282
- isLoading: false
283
- });
284
- }
285
- };
286
- fetchData();
287
1593
  }, [
288
- debouncedSearchText,
289
- isDirty,
290
- setAutocompleteState
1594
+ performSearch,
1595
+ query,
1596
+ setSearchText
291
1597
  ]);
292
1598
  return {
293
1599
  searchData,
@@ -295,16 +1601,17 @@ const useSearch = () => {
295
1601
  merchantShortName: safeMerchantShortName,
296
1602
  productCardConfig: safeProductCardConfig,
297
1603
  productList,
1604
+ recommendedProducts,
298
1605
  autocompleteResults,
299
1606
  searchFilters: filters,
300
1607
  availableDynamicFilters,
301
1608
  selectedFilterOptions,
302
1609
  searchText,
1610
+ query: query || "",
303
1611
  searchResultsState,
304
- isLoadingAutocomplete,
305
1612
  isLoadingSearch,
306
1613
  isFilterOpen,
307
- isDirty,
1614
+ shouldShowAutocomplete,
308
1615
  focusedIndex,
309
1616
  focusedOptionId,
310
1617
  filterButtonText,
@@ -312,6 +1619,8 @@ const useSearch = () => {
312
1619
  onSubmitSearch: handleSubmitSearch,
313
1620
  onAutocompleteSelect: handleAutocompleteSelect,
314
1621
  onKeyDown: handleKeyDown,
1622
+ onSearchInputFocus: handleSearchInputFocus,
1623
+ onSearchInputBlur: handleSearchInputBlur,
315
1624
  onToggleDynamicFilter: handleToggleDynamicFilter,
316
1625
  onSelectFilterItem: handleSelectFilterItem,
317
1626
  onRemoveFilter: handleRemoveFilter,
@@ -323,4 +1632,4 @@ const useSearch = () => {
323
1632
 
324
1633
  //#endregion
325
1634
  exports.useSearch = useSearch;
326
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.cjs","names":["useNewOrgConfig","orgShortNameAtom","searchAtom","filteredSearchProductsAtom","performSearchAtom","autocompleteStateAtom","searchParamsAtom","isFilterOpenAtom","searchSelectedFiltersAtom","addSearchFilterAtom","removeSearchFilterAtom","searchProductSortingAtom","clearSearchFiltersAtom","searchFiltersAtom","useDebounce","getSearchResultsState","formatFilterDisplayName","ProductSorting","useAmplitude","SpiffyMetricsEventName","createFilterOption","SpiffyWidgets","handleSelectFilterItem: SelectFilterItem","SearchResultsState"],"sources":["../../../src/hooks/Search/useSearch.tsx"],"sourcesContent":["import { useAtom, useAtomValue, useSetAtom } from \"jotai\";\nimport { useCallback, useEffect, useMemo, useRef, useState } from \"react\";\nimport { useDebounce } from \"src/hooks/Debounce/useDebounce\";\nimport {\n  addSearchFilterAtom,\n  clearSearchFiltersAtom,\n  createFilterOption,\n  filteredSearchProductsAtom,\n  performSearchAtom,\n  removeSearchFilterAtom,\n  searchAtom,\n  searchFiltersAtom,\n  searchParamsAtom,\n  searchProductSortingAtom,\n  searchSelectedFiltersAtom,\n  SelectedFilterOption,\n} from \"src/atoms/search\";\nimport {\n  autocompleteStateAtom,\n  isFilterOpenAtom,\n} from \"src/atoms/globalSearch/globalSearch\";\nimport { formatFilterDisplayName } from \"src/atoms/search/utils\";\nimport { ProductSorting } from \"src/atoms/search/types\";\nimport {\n  SpiffyMetricsEventName,\n  useAmplitude,\n} from \"src/contexts/amplitudeContext/amplitudeContext\";\nimport { SpiffyWidgets } from \"src/application/models/spiffyWidgets\";\nimport { ProductCardConfig } from \"src/contexts/types\";\nimport Logger from \"src/application/logging/logger\";\nimport { SearchResult } from \"src/application/models/api/search\";\nimport { SearchResponseProduct } from \"@spiffy-ai/commerce-api-client\";\nimport {\n  SearchFilterDatum,\n  SelectFilterItem,\n} from \"src/types/search-filter-types\";\nimport { getSearchResultsState, SearchResultsState } from \"../utils\";\nimport { orgShortNameAtom } from \"src/atoms/envive/enviveConfig\";\nimport { useNewOrgConfig } from \"../NewOrgConfig\";\nimport { useTrackComponentVisibleEvent } from \"../TrackComponentVisibleEvent\";\n\nexport interface SearchResultsHocProps {\n  // Data\n  searchData: SearchResult | null;\n  searchResponseId: string;\n  merchantShortName: string;\n  productCardConfig: ProductCardConfig;\n  productList: SearchResponseProduct[];\n  autocompleteResults: string[];\n  searchFilters: SearchFilterDatum[];\n  availableDynamicFilters: { name: string; displayName: string }[];\n  selectedFilterOptions: SelectedFilterOption[];\n\n  // State\n  searchText: string;\n  searchResultsState: SearchResultsState;\n  isLoadingAutocomplete: boolean;\n  isLoadingSearch: boolean;\n  isFilterOpen: boolean;\n  isDirty: boolean;\n  focusedIndex: number;\n  focusedOptionId: string | undefined;\n\n  // UI\n  filterButtonText: string;\n\n  // Event Handlers\n  onSearchInputChange: (value: string) => void;\n  onSubmitSearch: () => void;\n  onAutocompleteSelect: (suggestion: string) => void;\n  onKeyDown: (event: React.KeyboardEvent<HTMLInputElement>) => void;\n  onToggleDynamicFilter: ({\n    filter,\n    dynamicFilterDisplayName,\n  }: {\n    filter: string;\n    dynamicFilterDisplayName: string;\n  }) => void;\n  onSelectFilterItem: SelectFilterItem;\n  onRemoveFilter: (filter: SelectedFilterOption) => void;\n  onClearAllFilters: () => void;\n  setIsFilterOpen: (isFilterOpen: boolean) => void;\n\n  // Refs\n  searchResultsRef: React.RefObject<HTMLDivElement>;\n}\n\nexport const useSearch = (): SearchResultsHocProps => {\n  // Atoms\n  const config = useNewOrgConfig();\n  const orgShortName = useAtomValue(orgShortNameAtom);\n  const { data: searchData, loading: isLoadingSearch } =\n    useAtomValue(searchAtom);\n  const productList = useAtomValue(filteredSearchProductsAtom);\n  const performSearch = useSetAtom(performSearchAtom);\n  const [\n    { results: autocompleteResults, isLoading: isLoadingAutocomplete },\n    setAutocompleteState,\n  ] = useAtom(autocompleteStateAtom);\n  const [{ query }] = useAtom(searchParamsAtom);\n  const [isFilterOpen, setIsFilterOpen] = useAtom(isFilterOpenAtom);\n  const [selectedFilterOptions] = useAtom(searchSelectedFiltersAtom);\n  const addFilter = useSetAtom(addSearchFilterAtom);\n  const removeFilter = useSetAtom(removeSearchFilterAtom);\n  const [productSorting, setProductSorting] = useAtom(searchProductSortingAtom);\n  const clearFilters = useSetAtom(clearSearchFiltersAtom);\n  const searchFilters = useAtomValue(searchFiltersAtom);\n\n  // State\n  const [isDirty, setIsDirty] = useState(true);\n  const [focusedIndex, setFocusedIndex] = useState(-1);\n  const [focusedOptionId, setFocusedOptionId] = useState<string | undefined>(\n    undefined\n  );\n  const [searchText, setSearchText] = useState(query || \"\");\n\n  // Refs\n  const searchResultsRef = useRef<HTMLDivElement>(null);\n\n  // Derived State\n  const debouncedSearchText = useDebounce(searchText, 200);\n  const searchResultsState = getSearchResultsState(isLoadingSearch, searchData);\n\n  const dynamicFilters = searchData?.filters || [];\n\n  // Provide fallback values when orgUIConfig is not yet available\n  const safeProductCardConfig = config?.frontendConfig?.uiConfigs\n    ?.productCardConfig || {\n    variant: \"minimal\",\n    hoverVariant: \"none\",\n    layoutVariant: \"square\",\n  };\n  const safeMerchantShortName = orgShortName || \"\";\n\n  const availableDynamicFilters = useMemo(() => {\n    return dynamicFilters\n      .filter(\n        (dynamicFilterName) =>\n          !selectedFilterOptions.some(\n            (option) => option.id === `dynamic:${dynamicFilterName}`\n          )\n      )\n      .map((dynamicFilterName) => ({\n        name: dynamicFilterName,\n        displayName: formatFilterDisplayName(dynamicFilterName),\n      }));\n  }, [dynamicFilters, selectedFilterOptions]);\n\n  const filters = useMemo(() => {\n    const sortOptions = [\n      {\n        filterItemId: String(ProductSorting.FEATURED),\n        displayName: \"Relevance\",\n        productCount: 0,\n        isSelected: productSorting === ProductSorting.FEATURED,\n      },\n      {\n        filterItemId: String(ProductSorting.PRICE_ASC),\n        displayName: \"Price: Low to High\",\n        productCount: 0,\n        isSelected: productSorting === ProductSorting.PRICE_ASC,\n      },\n      {\n        filterItemId: String(ProductSorting.PRICE_DESC),\n        displayName: \"Price: High to Low\",\n        productCount: 0,\n        isSelected: productSorting === ProductSorting.PRICE_DESC,\n      },\n    ];\n\n    return [\n      { filterId: \"sort\", displayName: \"SORT\", items: sortOptions },\n      ...searchFilters,\n    ] as SearchFilterDatum[];\n  }, [productSorting, searchFilters]);\n\n  const filterButtonText = useMemo(() => {\n    const selectedCount = filters.reduce((acc: number, filter) => {\n      if (filter.filterId === \"sort\") {\n        return acc;\n      }\n      return acc + filter.items.filter((item) => item.isSelected).length;\n    }, 0);\n    if (selectedCount === 0) {\n      return \"Filter & Sort\";\n    }\n    return `Filter & Sort (${selectedCount})`;\n  }, [filters]);\n\n  // Callbacks\n  const { trackEvent } = useAmplitude();\n\n  const handleToggleDynamicFilter = useCallback(\n    ({\n      filter,\n      dynamicFilterDisplayName,\n    }: {\n      filter: string;\n      dynamicFilterDisplayName: string;\n    }) => {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.SearchFilterClicked,\n        eventProps: {\n          filterType: \"Dynamic\",\n          filterValue: filter,\n          queryText: searchText,\n        },\n      });\n      addFilter(\n        createFilterOption(\"dynamic\", filter, dynamicFilterDisplayName)\n      );\n    },\n    [addFilter, searchText, trackEvent]\n  );\n\n  const handleRemoveFilter = useCallback(\n    (filter: SelectedFilterOption) => {\n      removeFilter(filter.id);\n    },\n    [removeFilter]\n  );\n\n  const handleSubmitSearch = useCallback(async () => {\n    if (searchText.trim()) {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.SearchQuerySubmitted,\n        eventProps: {\n          searchOrigin: SpiffyWidgets.SearchResults,\n          queryText: searchText.trim(),\n        },\n        alsoSendToGoogleAnalytics: true,\n      });\n      const url = new URL(window.location.href);\n      url.searchParams.set(\"esq\", searchText.trim());\n      window.history.pushState({}, \"\", url);\n      performSearch({ query: searchText.trim() });\n    }\n  }, [performSearch, searchText, trackEvent]);\n\n  const handleAutocompleteSelect = useCallback(\n    (suggestion: string) => {\n      setSearchText(suggestion);\n      handleSubmitSearch();\n    },\n    [handleSubmitSearch, setSearchText]\n  );\n\n  const handleKeyDown = useCallback(\n    (event: React.KeyboardEvent<HTMLInputElement>) => {\n      if (event.key === \"ArrowDown\") {\n        event.preventDefault();\n        const newIndex = (focusedIndex + 1) % autocompleteResults.length;\n        setFocusedIndex(newIndex);\n        setFocusedOptionId(`option-${newIndex}`);\n      } else if (event.key === \"ArrowUp\") {\n        event.preventDefault();\n        const newIndex =\n          (focusedIndex - 1 + autocompleteResults.length) %\n          autocompleteResults.length;\n        setFocusedIndex(newIndex);\n        setFocusedOptionId(`option-${newIndex}`);\n      } else if (event.key === \"Enter\") {\n        if (focusedIndex === -1) {\n          event.preventDefault();\n          handleSubmitSearch();\n        } else {\n          event.preventDefault();\n          const suggestionText = autocompleteResults[focusedIndex];\n          handleAutocompleteSelect(suggestionText);\n        }\n      } else if (event.key === \"Escape\") {\n        event.preventDefault();\n        setFocusedIndex(-1);\n        setFocusedOptionId(undefined);\n      }\n    },\n    [\n      autocompleteResults,\n      focusedIndex,\n      handleAutocompleteSelect,\n      handleSubmitSearch,\n    ]\n  );\n\n  const handleSearchInputChange = (newValue: string) => {\n    if (newValue.length === 1) {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.SearchInputStarted,\n        eventProps: {\n          searchOrigin: SpiffyWidgets.SearchResults,\n        },\n      });\n    }\n    setSearchText(newValue);\n    setIsDirty(true);\n  };\n\n  const handleSelectFilterItem: SelectFilterItem = useCallback(\n    ({\n      filterId,\n      filterItemId,\n      isSelected,\n      displayName,\n    }: {\n      filterId: string;\n      filterItemId: string;\n      isSelected: boolean;\n      displayName: string;\n    }) => {\n      if (filterId === \"sort\") {\n        const newSort = filterItemId as ProductSorting;\n        trackEvent({\n          eventName: SpiffyMetricsEventName.SearchSortClicked,\n          eventProps: {\n            sortType: newSort,\n            queryText: searchText,\n          },\n        });\n        setProductSorting(newSort);\n      } else if (!isSelected) {\n        removeFilter(`${filterId}:${filterItemId}`);\n      } else {\n        trackEvent({\n          eventName: SpiffyMetricsEventName.SearchFilterClicked,\n          eventProps: {\n            filterType: \"Static\",\n            filterCategory: filterId,\n            filterValue: filterItemId,\n            queryText: searchText,\n          },\n        });\n        addFilter(createFilterOption(filterId, filterItemId, displayName));\n      }\n    },\n    [addFilter, removeFilter, setProductSorting, searchText, trackEvent]\n  );\n\n  const handleClearAllFilters = useCallback(() => {\n    setProductSorting(ProductSorting.FEATURED);\n    clearFilters();\n  }, [setProductSorting, clearFilters]);\n\n  // Side Effects\n  useTrackComponentVisibleEvent(\n    SpiffyWidgets.SearchResults,\n    searchResultsRef,\n    {},\n    SpiffyMetricsEventName.SearchComponentVisible\n  );\n\n  useEffect(() => {\n    if (searchResultsState === SearchResultsState.NoResults || searchResultsState === SearchResultsState.Results) {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.SearchResultsViewed,\n        eventProps: {\n          queryText: searchText,\n          resultsCount: productList.length,\n        },\n      });\n    }\n  }, [productList.length, searchResultsState, trackEvent]);\n\n  useEffect(() => {\n    if (query && query !== searchText) {\n      setSearchText(query);\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [query]);\n\n  useEffect(() => {\n    const esq = new URLSearchParams(window.location.search).get(\"esq\");\n    if (esq) {\n      setSearchText(esq);\n      performSearch({ query: esq });\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [performSearch]);\n\n  const fetchAutocompleteSuggestions = (_query: string) => {\n    // TODO: implement autocomplete suggestions\n    return Promise.resolve([]);\n  };\n\n  useEffect(() => {\n    if (fetchAutocompleteSuggestions === undefined) {\n      return;\n    }\n\n    if (!isDirty || debouncedSearchText.length <= 2) {\n      setAutocompleteState({ results: [], isLoading: false });\n      return;\n    }\n\n    setAutocompleteState((prev) => ({ ...prev, isLoading: true }));\n\n    const fetchData = async () => {\n      try {\n        const results = await fetchAutocompleteSuggestions?.(\n          debouncedSearchText\n        );\n        setAutocompleteState({ results: results ?? [], isLoading: false });\n      } catch (error) {\n        Logger.logError(\"Failed to fetch autocomplete suggestions:\", error);\n        setAutocompleteState({ results: [], isLoading: false });\n      }\n    };\n\n    fetchData();\n  }, [debouncedSearchText, isDirty, setAutocompleteState]);\n\n  return {\n    searchData,\n    searchResponseId: searchData?.searchResponseId ?? \"\",\n    merchantShortName: safeMerchantShortName,\n    productCardConfig: safeProductCardConfig,\n    productList,\n    autocompleteResults,\n    searchFilters: filters,\n    availableDynamicFilters,\n    selectedFilterOptions,\n    searchText,\n    searchResultsState,\n    isLoadingAutocomplete,\n    isLoadingSearch,\n    isFilterOpen,\n    isDirty,\n    focusedIndex,\n    focusedOptionId,\n    filterButtonText,\n    onSearchInputChange: handleSearchInputChange,\n    onSubmitSearch: handleSubmitSearch,\n    onAutocompleteSelect: handleAutocompleteSelect,\n    onKeyDown: handleKeyDown,\n    onToggleDynamicFilter: handleToggleDynamicFilter,\n    onSelectFilterItem: handleSelectFilterItem,\n    onRemoveFilter: handleRemoveFilter,\n    onClearAllFilters: handleClearAllFilters,\n    setIsFilterOpen,\n    searchResultsRef,\n  };\n};\n"],"mappings":"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"}
1635
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.cjs","names":["obj","path","norm","value","pattern","result","item","searchers","query","amplitudeTrackEventAtom","autocompleteStateAtom","SpiffyMetricsEventName","productRetrievalFunction:\n  | ((params: ProductRetrievalParams) => Promise<ProductRetrievalResult>)\n  | null","get","error: unknown","useFeatureFlagService","FeatureGates","useNewOrgConfig","useNewOrgConfig","orgShortNameAtom","searchAtom","filteredSearchProductsAtom","performSearchAtom","searchParamsAtom","isFilterOpenAtom","searchSelectedFiltersAtom","addSearchFilterAtom","removeSearchFilterAtom","searchProductSortingAtom","clearSearchFiltersAtom","searchFiltersAtom","SpiffyWidgets","getSearchResultsState","formatFilterDisplayName","ProductSorting","useAmplitude","SpiffyMetricsEventName","createFilterOption","handleSelectFilterItem: SelectFilterItem","SearchResultsState"],"sources":["../../../node_modules/fuse.js/dist/fuse.mjs","../../../src/hooks/Search/useSearchInput.ts","../../../src/atoms/search/productRetrievalAdapter.ts","../../../src/atoms/search/productRetrievalAPI.ts","../../../src/hooks/Search/useRecommendedProducts.ts","../../../src/hooks/Search/useSearch.tsx"],"sourcesContent":["/**\n * Fuse.js v7.1.0 - Lightweight fuzzy-search (http://fusejs.io)\n *\n * Copyright (c) 2025 Kiro Risk (http://kiro.me)\n * All Rights Reserved. Apache Software License 2.0\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n */\n\nfunction isArray(value) {\n  return !Array.isArray\n    ? getTag(value) === '[object Array]'\n    : Array.isArray(value)\n}\n\n// Adapted from: https://github.com/lodash/lodash/blob/master/.internal/baseToString.js\nconst INFINITY = 1 / 0;\nfunction baseToString(value) {\n  // Exit early for strings to avoid a performance hit in some environments.\n  if (typeof value == 'string') {\n    return value\n  }\n  let result = value + '';\n  return result == '0' && 1 / value == -INFINITY ? '-0' : result\n}\n\nfunction toString(value) {\n  return value == null ? '' : baseToString(value)\n}\n\nfunction isString(value) {\n  return typeof value === 'string'\n}\n\nfunction isNumber(value) {\n  return typeof value === 'number'\n}\n\n// Adapted from: https://github.com/lodash/lodash/blob/master/isBoolean.js\nfunction isBoolean(value) {\n  return (\n    value === true ||\n    value === false ||\n    (isObjectLike(value) && getTag(value) == '[object Boolean]')\n  )\n}\n\nfunction isObject(value) {\n  return typeof value === 'object'\n}\n\n// Checks if `value` is object-like.\nfunction isObjectLike(value) {\n  return isObject(value) && value !== null\n}\n\nfunction isDefined(value) {\n  return value !== undefined && value !== null\n}\n\nfunction isBlank(value) {\n  return !value.trim().length\n}\n\n// Gets the `toStringTag` of `value`.\n// Adapted from: https://github.com/lodash/lodash/blob/master/.internal/getTag.js\nfunction getTag(value) {\n  return value == null\n    ? value === undefined\n      ? '[object Undefined]'\n      : '[object Null]'\n    : Object.prototype.toString.call(value)\n}\n\nconst EXTENDED_SEARCH_UNAVAILABLE = 'Extended search is not available';\n\nconst INCORRECT_INDEX_TYPE = \"Incorrect 'index' type\";\n\nconst LOGICAL_SEARCH_INVALID_QUERY_FOR_KEY = (key) =>\n  `Invalid value for key ${key}`;\n\nconst PATTERN_LENGTH_TOO_LARGE = (max) =>\n  `Pattern length exceeds max of ${max}.`;\n\nconst MISSING_KEY_PROPERTY = (name) => `Missing ${name} property in key`;\n\nconst INVALID_KEY_WEIGHT_VALUE = (key) =>\n  `Property 'weight' in key '${key}' must be a positive integer`;\n\nconst hasOwn = Object.prototype.hasOwnProperty;\n\nclass KeyStore {\n  constructor(keys) {\n    this._keys = [];\n    this._keyMap = {};\n\n    let totalWeight = 0;\n\n    keys.forEach((key) => {\n      let obj = createKey(key);\n\n      this._keys.push(obj);\n      this._keyMap[obj.id] = obj;\n\n      totalWeight += obj.weight;\n    });\n\n    // Normalize weights so that their sum is equal to 1\n    this._keys.forEach((key) => {\n      key.weight /= totalWeight;\n    });\n  }\n  get(keyId) {\n    return this._keyMap[keyId]\n  }\n  keys() {\n    return this._keys\n  }\n  toJSON() {\n    return JSON.stringify(this._keys)\n  }\n}\n\nfunction createKey(key) {\n  let path = null;\n  let id = null;\n  let src = null;\n  let weight = 1;\n  let getFn = null;\n\n  if (isString(key) || isArray(key)) {\n    src = key;\n    path = createKeyPath(key);\n    id = createKeyId(key);\n  } else {\n    if (!hasOwn.call(key, 'name')) {\n      throw new Error(MISSING_KEY_PROPERTY('name'))\n    }\n\n    const name = key.name;\n    src = name;\n\n    if (hasOwn.call(key, 'weight')) {\n      weight = key.weight;\n\n      if (weight <= 0) {\n        throw new Error(INVALID_KEY_WEIGHT_VALUE(name))\n      }\n    }\n\n    path = createKeyPath(name);\n    id = createKeyId(name);\n    getFn = key.getFn;\n  }\n\n  return { path, id, weight, src, getFn }\n}\n\nfunction createKeyPath(key) {\n  return isArray(key) ? key : key.split('.')\n}\n\nfunction createKeyId(key) {\n  return isArray(key) ? key.join('.') : key\n}\n\nfunction get(obj, path) {\n  let list = [];\n  let arr = false;\n\n  const deepGet = (obj, path, index) => {\n    if (!isDefined(obj)) {\n      return\n    }\n    if (!path[index]) {\n      // If there's no path left, we've arrived at the object we care about.\n      list.push(obj);\n    } else {\n      let key = path[index];\n\n      const value = obj[key];\n\n      if (!isDefined(value)) {\n        return\n      }\n\n      // If we're at the last value in the path, and if it's a string/number/bool,\n      // add it to the list\n      if (\n        index === path.length - 1 &&\n        (isString(value) || isNumber(value) || isBoolean(value))\n      ) {\n        list.push(toString(value));\n      } else if (isArray(value)) {\n        arr = true;\n        // Search each item in the array.\n        for (let i = 0, len = value.length; i < len; i += 1) {\n          deepGet(value[i], path, index + 1);\n        }\n      } else if (path.length) {\n        // An object. Recurse further.\n        deepGet(value, path, index + 1);\n      }\n    }\n  };\n\n  // Backwards compatibility (since path used to be a string)\n  deepGet(obj, isString(path) ? path.split('.') : path, 0);\n\n  return arr ? list : list[0]\n}\n\nconst MatchOptions = {\n  // Whether the matches should be included in the result set. When `true`, each record in the result\n  // set will include the indices of the matched characters.\n  // These can consequently be used for highlighting purposes.\n  includeMatches: false,\n  // When `true`, the matching function will continue to the end of a search pattern even if\n  // a perfect match has already been located in the string.\n  findAllMatches: false,\n  // Minimum number of characters that must be matched before a result is considered a match\n  minMatchCharLength: 1\n};\n\nconst BasicOptions = {\n  // When `true`, the algorithm continues searching to the end of the input even if a perfect\n  // match is found before the end of the same input.\n  isCaseSensitive: false,\n  // When `true`, the algorithm will ignore diacritics (accents) in comparisons\n  ignoreDiacritics: false,\n  // When true, the matching function will continue to the end of a search pattern even if\n  includeScore: false,\n  // List of properties that will be searched. This also supports nested properties.\n  keys: [],\n  // Whether to sort the result list, by score\n  shouldSort: true,\n  // Default sort function: sort by ascending score, ascending index\n  sortFn: (a, b) =>\n    a.score === b.score ? (a.idx < b.idx ? -1 : 1) : a.score < b.score ? -1 : 1\n};\n\nconst FuzzyOptions = {\n  // Approximately where in the text is the pattern expected to be found?\n  location: 0,\n  // At what point does the match algorithm give up. A threshold of '0.0' requires a perfect match\n  // (of both letters and location), a threshold of '1.0' would match anything.\n  threshold: 0.6,\n  // Determines how close the match must be to the fuzzy location (specified above).\n  // An exact letter match which is 'distance' characters away from the fuzzy location\n  // would score as a complete mismatch. A distance of '0' requires the match be at\n  // the exact location specified, a threshold of '1000' would require a perfect match\n  // to be within 800 characters of the fuzzy location to be found using a 0.8 threshold.\n  distance: 100\n};\n\nconst AdvancedOptions = {\n  // When `true`, it enables the use of unix-like search commands\n  useExtendedSearch: false,\n  // The get function to use when fetching an object's properties.\n  // The default will search nested paths *ie foo.bar.baz*\n  getFn: get,\n  // When `true`, search will ignore `location` and `distance`, so it won't matter\n  // where in the string the pattern appears.\n  // More info: https://fusejs.io/concepts/scoring-theory.html#fuzziness-score\n  ignoreLocation: false,\n  // When `true`, the calculation for the relevance score (used for sorting) will\n  // ignore the field-length norm.\n  // More info: https://fusejs.io/concepts/scoring-theory.html#field-length-norm\n  ignoreFieldNorm: false,\n  // The weight to determine how much field length norm effects scoring.\n  fieldNormWeight: 1\n};\n\nvar Config = {\n  ...BasicOptions,\n  ...MatchOptions,\n  ...FuzzyOptions,\n  ...AdvancedOptions\n};\n\nconst SPACE = /[^ ]+/g;\n\n// Field-length norm: the shorter the field, the higher the weight.\n// Set to 3 decimals to reduce index size.\nfunction norm(weight = 1, mantissa = 3) {\n  const cache = new Map();\n  const m = Math.pow(10, mantissa);\n\n  return {\n    get(value) {\n      const numTokens = value.match(SPACE).length;\n\n      if (cache.has(numTokens)) {\n        return cache.get(numTokens)\n      }\n\n      // Default function is 1/sqrt(x), weight makes that variable\n      const norm = 1 / Math.pow(numTokens, 0.5 * weight);\n\n      // In place of `toFixed(mantissa)`, for faster computation\n      const n = parseFloat(Math.round(norm * m) / m);\n\n      cache.set(numTokens, n);\n\n      return n\n    },\n    clear() {\n      cache.clear();\n    }\n  }\n}\n\nclass FuseIndex {\n  constructor({\n    getFn = Config.getFn,\n    fieldNormWeight = Config.fieldNormWeight\n  } = {}) {\n    this.norm = norm(fieldNormWeight, 3);\n    this.getFn = getFn;\n    this.isCreated = false;\n\n    this.setIndexRecords();\n  }\n  setSources(docs = []) {\n    this.docs = docs;\n  }\n  setIndexRecords(records = []) {\n    this.records = records;\n  }\n  setKeys(keys = []) {\n    this.keys = keys;\n    this._keysMap = {};\n    keys.forEach((key, idx) => {\n      this._keysMap[key.id] = idx;\n    });\n  }\n  create() {\n    if (this.isCreated || !this.docs.length) {\n      return\n    }\n\n    this.isCreated = true;\n\n    // List is Array<String>\n    if (isString(this.docs[0])) {\n      this.docs.forEach((doc, docIndex) => {\n        this._addString(doc, docIndex);\n      });\n    } else {\n      // List is Array<Object>\n      this.docs.forEach((doc, docIndex) => {\n        this._addObject(doc, docIndex);\n      });\n    }\n\n    this.norm.clear();\n  }\n  // Adds a doc to the end of the index\n  add(doc) {\n    const idx = this.size();\n\n    if (isString(doc)) {\n      this._addString(doc, idx);\n    } else {\n      this._addObject(doc, idx);\n    }\n  }\n  // Removes the doc at the specified index of the index\n  removeAt(idx) {\n    this.records.splice(idx, 1);\n\n    // Change ref index of every subsquent doc\n    for (let i = idx, len = this.size(); i < len; i += 1) {\n      this.records[i].i -= 1;\n    }\n  }\n  getValueForItemAtKeyId(item, keyId) {\n    return item[this._keysMap[keyId]]\n  }\n  size() {\n    return this.records.length\n  }\n  _addString(doc, docIndex) {\n    if (!isDefined(doc) || isBlank(doc)) {\n      return\n    }\n\n    let record = {\n      v: doc,\n      i: docIndex,\n      n: this.norm.get(doc)\n    };\n\n    this.records.push(record);\n  }\n  _addObject(doc, docIndex) {\n    let record = { i: docIndex, $: {} };\n\n    // Iterate over every key (i.e, path), and fetch the value at that key\n    this.keys.forEach((key, keyIndex) => {\n      let value = key.getFn ? key.getFn(doc) : this.getFn(doc, key.path);\n\n      if (!isDefined(value)) {\n        return\n      }\n\n      if (isArray(value)) {\n        let subRecords = [];\n        const stack = [{ nestedArrIndex: -1, value }];\n\n        while (stack.length) {\n          const { nestedArrIndex, value } = stack.pop();\n\n          if (!isDefined(value)) {\n            continue\n          }\n\n          if (isString(value) && !isBlank(value)) {\n            let subRecord = {\n              v: value,\n              i: nestedArrIndex,\n              n: this.norm.get(value)\n            };\n\n            subRecords.push(subRecord);\n          } else if (isArray(value)) {\n            value.forEach((item, k) => {\n              stack.push({\n                nestedArrIndex: k,\n                value: item\n              });\n            });\n          } else ;\n        }\n        record.$[keyIndex] = subRecords;\n      } else if (isString(value) && !isBlank(value)) {\n        let subRecord = {\n          v: value,\n          n: this.norm.get(value)\n        };\n\n        record.$[keyIndex] = subRecord;\n      }\n    });\n\n    this.records.push(record);\n  }\n  toJSON() {\n    return {\n      keys: this.keys,\n      records: this.records\n    }\n  }\n}\n\nfunction createIndex(\n  keys,\n  docs,\n  { getFn = Config.getFn, fieldNormWeight = Config.fieldNormWeight } = {}\n) {\n  const myIndex = new FuseIndex({ getFn, fieldNormWeight });\n  myIndex.setKeys(keys.map(createKey));\n  myIndex.setSources(docs);\n  myIndex.create();\n  return myIndex\n}\n\nfunction parseIndex(\n  data,\n  { getFn = Config.getFn, fieldNormWeight = Config.fieldNormWeight } = {}\n) {\n  const { keys, records } = data;\n  const myIndex = new FuseIndex({ getFn, fieldNormWeight });\n  myIndex.setKeys(keys);\n  myIndex.setIndexRecords(records);\n  return myIndex\n}\n\nfunction computeScore$1(\n  pattern,\n  {\n    errors = 0,\n    currentLocation = 0,\n    expectedLocation = 0,\n    distance = Config.distance,\n    ignoreLocation = Config.ignoreLocation\n  } = {}\n) {\n  const accuracy = errors / pattern.length;\n\n  if (ignoreLocation) {\n    return accuracy\n  }\n\n  const proximity = Math.abs(expectedLocation - currentLocation);\n\n  if (!distance) {\n    // Dodge divide by zero error.\n    return proximity ? 1.0 : accuracy\n  }\n\n  return accuracy + proximity / distance\n}\n\nfunction convertMaskToIndices(\n  matchmask = [],\n  minMatchCharLength = Config.minMatchCharLength\n) {\n  let indices = [];\n  let start = -1;\n  let end = -1;\n  let i = 0;\n\n  for (let len = matchmask.length; i < len; i += 1) {\n    let match = matchmask[i];\n    if (match && start === -1) {\n      start = i;\n    } else if (!match && start !== -1) {\n      end = i - 1;\n      if (end - start + 1 >= minMatchCharLength) {\n        indices.push([start, end]);\n      }\n      start = -1;\n    }\n  }\n\n  // (i-1 - start) + 1 => i - start\n  if (matchmask[i - 1] && i - start >= minMatchCharLength) {\n    indices.push([start, i - 1]);\n  }\n\n  return indices\n}\n\n// Machine word size\nconst MAX_BITS = 32;\n\nfunction search(\n  text,\n  pattern,\n  patternAlphabet,\n  {\n    location = Config.location,\n    distance = Config.distance,\n    threshold = Config.threshold,\n    findAllMatches = Config.findAllMatches,\n    minMatchCharLength = Config.minMatchCharLength,\n    includeMatches = Config.includeMatches,\n    ignoreLocation = Config.ignoreLocation\n  } = {}\n) {\n  if (pattern.length > MAX_BITS) {\n    throw new Error(PATTERN_LENGTH_TOO_LARGE(MAX_BITS))\n  }\n\n  const patternLen = pattern.length;\n  // Set starting location at beginning text and initialize the alphabet.\n  const textLen = text.length;\n  // Handle the case when location > text.length\n  const expectedLocation = Math.max(0, Math.min(location, textLen));\n  // Highest score beyond which we give up.\n  let currentThreshold = threshold;\n  // Is there a nearby exact match? (speedup)\n  let bestLocation = expectedLocation;\n\n  // Performance: only computer matches when the minMatchCharLength > 1\n  // OR if `includeMatches` is true.\n  const computeMatches = minMatchCharLength > 1 || includeMatches;\n  // A mask of the matches, used for building the indices\n  const matchMask = computeMatches ? Array(textLen) : [];\n\n  let index;\n\n  // Get all exact matches, here for speed up\n  while ((index = text.indexOf(pattern, bestLocation)) > -1) {\n    let score = computeScore$1(pattern, {\n      currentLocation: index,\n      expectedLocation,\n      distance,\n      ignoreLocation\n    });\n\n    currentThreshold = Math.min(score, currentThreshold);\n    bestLocation = index + patternLen;\n\n    if (computeMatches) {\n      let i = 0;\n      while (i < patternLen) {\n        matchMask[index + i] = 1;\n        i += 1;\n      }\n    }\n  }\n\n  // Reset the best location\n  bestLocation = -1;\n\n  let lastBitArr = [];\n  let finalScore = 1;\n  let binMax = patternLen + textLen;\n\n  const mask = 1 << (patternLen - 1);\n\n  for (let i = 0; i < patternLen; i += 1) {\n    // Scan for the best match; each iteration allows for one more error.\n    // Run a binary search to determine how far from the match location we can stray\n    // at this error level.\n    let binMin = 0;\n    let binMid = binMax;\n\n    while (binMin < binMid) {\n      const score = computeScore$1(pattern, {\n        errors: i,\n        currentLocation: expectedLocation + binMid,\n        expectedLocation,\n        distance,\n        ignoreLocation\n      });\n\n      if (score <= currentThreshold) {\n        binMin = binMid;\n      } else {\n        binMax = binMid;\n      }\n\n      binMid = Math.floor((binMax - binMin) / 2 + binMin);\n    }\n\n    // Use the result from this iteration as the maximum for the next.\n    binMax = binMid;\n\n    let start = Math.max(1, expectedLocation - binMid + 1);\n    let finish = findAllMatches\n      ? textLen\n      : Math.min(expectedLocation + binMid, textLen) + patternLen;\n\n    // Initialize the bit array\n    let bitArr = Array(finish + 2);\n\n    bitArr[finish + 1] = (1 << i) - 1;\n\n    for (let j = finish; j >= start; j -= 1) {\n      let currentLocation = j - 1;\n      let charMatch = patternAlphabet[text.charAt(currentLocation)];\n\n      if (computeMatches) {\n        // Speed up: quick bool to int conversion (i.e, `charMatch ? 1 : 0`)\n        matchMask[currentLocation] = +!!charMatch;\n      }\n\n      // First pass: exact match\n      bitArr[j] = ((bitArr[j + 1] << 1) | 1) & charMatch;\n\n      // Subsequent passes: fuzzy match\n      if (i) {\n        bitArr[j] |=\n          ((lastBitArr[j + 1] | lastBitArr[j]) << 1) | 1 | lastBitArr[j + 1];\n      }\n\n      if (bitArr[j] & mask) {\n        finalScore = computeScore$1(pattern, {\n          errors: i,\n          currentLocation,\n          expectedLocation,\n          distance,\n          ignoreLocation\n        });\n\n        // This match will almost certainly be better than any existing match.\n        // But check anyway.\n        if (finalScore <= currentThreshold) {\n          // Indeed it is\n          currentThreshold = finalScore;\n          bestLocation = currentLocation;\n\n          // Already passed `loc`, downhill from here on in.\n          if (bestLocation <= expectedLocation) {\n            break\n          }\n\n          // When passing `bestLocation`, don't exceed our current distance from `expectedLocation`.\n          start = Math.max(1, 2 * expectedLocation - bestLocation);\n        }\n      }\n    }\n\n    // No hope for a (better) match at greater error levels.\n    const score = computeScore$1(pattern, {\n      errors: i + 1,\n      currentLocation: expectedLocation,\n      expectedLocation,\n      distance,\n      ignoreLocation\n    });\n\n    if (score > currentThreshold) {\n      break\n    }\n\n    lastBitArr = bitArr;\n  }\n\n  const result = {\n    isMatch: bestLocation >= 0,\n    // Count exact matches (those with a score of 0) to be \"almost\" exact\n    score: Math.max(0.001, finalScore)\n  };\n\n  if (computeMatches) {\n    const indices = convertMaskToIndices(matchMask, minMatchCharLength);\n    if (!indices.length) {\n      result.isMatch = false;\n    } else if (includeMatches) {\n      result.indices = indices;\n    }\n  }\n\n  return result\n}\n\nfunction createPatternAlphabet(pattern) {\n  let mask = {};\n\n  for (let i = 0, len = pattern.length; i < len; i += 1) {\n    const char = pattern.charAt(i);\n    mask[char] = (mask[char] || 0) | (1 << (len - i - 1));\n  }\n\n  return mask\n}\n\nconst stripDiacritics = String.prototype.normalize\n    ? ((str) => str.normalize('NFD').replace(/[\\u0300-\\u036F\\u0483-\\u0489\\u0591-\\u05BD\\u05BF\\u05C1\\u05C2\\u05C4\\u05C5\\u05C7\\u0610-\\u061A\\u064B-\\u065F\\u0670\\u06D6-\\u06DC\\u06DF-\\u06E4\\u06E7\\u06E8\\u06EA-\\u06ED\\u0711\\u0730-\\u074A\\u07A6-\\u07B0\\u07EB-\\u07F3\\u07FD\\u0816-\\u0819\\u081B-\\u0823\\u0825-\\u0827\\u0829-\\u082D\\u0859-\\u085B\\u08D3-\\u08E1\\u08E3-\\u0903\\u093A-\\u093C\\u093E-\\u094F\\u0951-\\u0957\\u0962\\u0963\\u0981-\\u0983\\u09BC\\u09BE-\\u09C4\\u09C7\\u09C8\\u09CB-\\u09CD\\u09D7\\u09E2\\u09E3\\u09FE\\u0A01-\\u0A03\\u0A3C\\u0A3E-\\u0A42\\u0A47\\u0A48\\u0A4B-\\u0A4D\\u0A51\\u0A70\\u0A71\\u0A75\\u0A81-\\u0A83\\u0ABC\\u0ABE-\\u0AC5\\u0AC7-\\u0AC9\\u0ACB-\\u0ACD\\u0AE2\\u0AE3\\u0AFA-\\u0AFF\\u0B01-\\u0B03\\u0B3C\\u0B3E-\\u0B44\\u0B47\\u0B48\\u0B4B-\\u0B4D\\u0B56\\u0B57\\u0B62\\u0B63\\u0B82\\u0BBE-\\u0BC2\\u0BC6-\\u0BC8\\u0BCA-\\u0BCD\\u0BD7\\u0C00-\\u0C04\\u0C3E-\\u0C44\\u0C46-\\u0C48\\u0C4A-\\u0C4D\\u0C55\\u0C56\\u0C62\\u0C63\\u0C81-\\u0C83\\u0CBC\\u0CBE-\\u0CC4\\u0CC6-\\u0CC8\\u0CCA-\\u0CCD\\u0CD5\\u0CD6\\u0CE2\\u0CE3\\u0D00-\\u0D03\\u0D3B\\u0D3C\\u0D3E-\\u0D44\\u0D46-\\u0D48\\u0D4A-\\u0D4D\\u0D57\\u0D62\\u0D63\\u0D82\\u0D83\\u0DCA\\u0DCF-\\u0DD4\\u0DD6\\u0DD8-\\u0DDF\\u0DF2\\u0DF3\\u0E31\\u0E34-\\u0E3A\\u0E47-\\u0E4E\\u0EB1\\u0EB4-\\u0EB9\\u0EBB\\u0EBC\\u0EC8-\\u0ECD\\u0F18\\u0F19\\u0F35\\u0F37\\u0F39\\u0F3E\\u0F3F\\u0F71-\\u0F84\\u0F86\\u0F87\\u0F8D-\\u0F97\\u0F99-\\u0FBC\\u0FC6\\u102B-\\u103E\\u1056-\\u1059\\u105E-\\u1060\\u1062-\\u1064\\u1067-\\u106D\\u1071-\\u1074\\u1082-\\u108D\\u108F\\u109A-\\u109D\\u135D-\\u135F\\u1712-\\u1714\\u1732-\\u1734\\u1752\\u1753\\u1772\\u1773\\u17B4-\\u17D3\\u17DD\\u180B-\\u180D\\u1885\\u1886\\u18A9\\u1920-\\u192B\\u1930-\\u193B\\u1A17-\\u1A1B\\u1A55-\\u1A5E\\u1A60-\\u1A7C\\u1A7F\\u1AB0-\\u1ABE\\u1B00-\\u1B04\\u1B34-\\u1B44\\u1B6B-\\u1B73\\u1B80-\\u1B82\\u1BA1-\\u1BAD\\u1BE6-\\u1BF3\\u1C24-\\u1C37\\u1CD0-\\u1CD2\\u1CD4-\\u1CE8\\u1CED\\u1CF2-\\u1CF4\\u1CF7-\\u1CF9\\u1DC0-\\u1DF9\\u1DFB-\\u1DFF\\u20D0-\\u20F0\\u2CEF-\\u2CF1\\u2D7F\\u2DE0-\\u2DFF\\u302A-\\u302F\\u3099\\u309A\\uA66F-\\uA672\\uA674-\\uA67D\\uA69E\\uA69F\\uA6F0\\uA6F1\\uA802\\uA806\\uA80B\\uA823-\\uA827\\uA880\\uA881\\uA8B4-\\uA8C5\\uA8E0-\\uA8F1\\uA8FF\\uA926-\\uA92D\\uA947-\\uA953\\uA980-\\uA983\\uA9B3-\\uA9C0\\uA9E5\\uAA29-\\uAA36\\uAA43\\uAA4C\\uAA4D\\uAA7B-\\uAA7D\\uAAB0\\uAAB2-\\uAAB4\\uAAB7\\uAAB8\\uAABE\\uAABF\\uAAC1\\uAAEB-\\uAAEF\\uAAF5\\uAAF6\\uABE3-\\uABEA\\uABEC\\uABED\\uFB1E\\uFE00-\\uFE0F\\uFE20-\\uFE2F]/g, ''))\n    : ((str) => str);\n\nclass BitapSearch {\n  constructor(\n    pattern,\n    {\n      location = Config.location,\n      threshold = Config.threshold,\n      distance = Config.distance,\n      includeMatches = Config.includeMatches,\n      findAllMatches = Config.findAllMatches,\n      minMatchCharLength = Config.minMatchCharLength,\n      isCaseSensitive = Config.isCaseSensitive,\n      ignoreDiacritics = Config.ignoreDiacritics,\n      ignoreLocation = Config.ignoreLocation\n    } = {}\n  ) {\n    this.options = {\n      location,\n      threshold,\n      distance,\n      includeMatches,\n      findAllMatches,\n      minMatchCharLength,\n      isCaseSensitive,\n      ignoreDiacritics,\n      ignoreLocation\n    };\n\n    pattern = isCaseSensitive ? pattern : pattern.toLowerCase();\n    pattern = ignoreDiacritics ? stripDiacritics(pattern) : pattern;\n    this.pattern = pattern;\n\n    this.chunks = [];\n\n    if (!this.pattern.length) {\n      return\n    }\n\n    const addChunk = (pattern, startIndex) => {\n      this.chunks.push({\n        pattern,\n        alphabet: createPatternAlphabet(pattern),\n        startIndex\n      });\n    };\n\n    const len = this.pattern.length;\n\n    if (len > MAX_BITS) {\n      let i = 0;\n      const remainder = len % MAX_BITS;\n      const end = len - remainder;\n\n      while (i < end) {\n        addChunk(this.pattern.substr(i, MAX_BITS), i);\n        i += MAX_BITS;\n      }\n\n      if (remainder) {\n        const startIndex = len - MAX_BITS;\n        addChunk(this.pattern.substr(startIndex), startIndex);\n      }\n    } else {\n      addChunk(this.pattern, 0);\n    }\n  }\n\n  searchIn(text) {\n    const { isCaseSensitive, ignoreDiacritics, includeMatches } = this.options;\n\n    text = isCaseSensitive ? text : text.toLowerCase();\n    text = ignoreDiacritics ? stripDiacritics(text) : text;\n\n    // Exact match\n    if (this.pattern === text) {\n      let result = {\n        isMatch: true,\n        score: 0\n      };\n\n      if (includeMatches) {\n        result.indices = [[0, text.length - 1]];\n      }\n\n      return result\n    }\n\n    // Otherwise, use Bitap algorithm\n    const {\n      location,\n      distance,\n      threshold,\n      findAllMatches,\n      minMatchCharLength,\n      ignoreLocation\n    } = this.options;\n\n    let allIndices = [];\n    let totalScore = 0;\n    let hasMatches = false;\n\n    this.chunks.forEach(({ pattern, alphabet, startIndex }) => {\n      const { isMatch, score, indices } = search(text, pattern, alphabet, {\n        location: location + startIndex,\n        distance,\n        threshold,\n        findAllMatches,\n        minMatchCharLength,\n        includeMatches,\n        ignoreLocation\n      });\n\n      if (isMatch) {\n        hasMatches = true;\n      }\n\n      totalScore += score;\n\n      if (isMatch && indices) {\n        allIndices = [...allIndices, ...indices];\n      }\n    });\n\n    let result = {\n      isMatch: hasMatches,\n      score: hasMatches ? totalScore / this.chunks.length : 1\n    };\n\n    if (hasMatches && includeMatches) {\n      result.indices = allIndices;\n    }\n\n    return result\n  }\n}\n\nclass BaseMatch {\n  constructor(pattern) {\n    this.pattern = pattern;\n  }\n  static isMultiMatch(pattern) {\n    return getMatch(pattern, this.multiRegex)\n  }\n  static isSingleMatch(pattern) {\n    return getMatch(pattern, this.singleRegex)\n  }\n  search(/*text*/) {}\n}\n\nfunction getMatch(pattern, exp) {\n  const matches = pattern.match(exp);\n  return matches ? matches[1] : null\n}\n\n// Token: 'file\n\nclass ExactMatch extends BaseMatch {\n  constructor(pattern) {\n    super(pattern);\n  }\n  static get type() {\n    return 'exact'\n  }\n  static get multiRegex() {\n    return /^=\"(.*)\"$/\n  }\n  static get singleRegex() {\n    return /^=(.*)$/\n  }\n  search(text) {\n    const isMatch = text === this.pattern;\n\n    return {\n      isMatch,\n      score: isMatch ? 0 : 1,\n      indices: [0, this.pattern.length - 1]\n    }\n  }\n}\n\n// Token: !fire\n\nclass InverseExactMatch extends BaseMatch {\n  constructor(pattern) {\n    super(pattern);\n  }\n  static get type() {\n    return 'inverse-exact'\n  }\n  static get multiRegex() {\n    return /^!\"(.*)\"$/\n  }\n  static get singleRegex() {\n    return /^!(.*)$/\n  }\n  search(text) {\n    const index = text.indexOf(this.pattern);\n    const isMatch = index === -1;\n\n    return {\n      isMatch,\n      score: isMatch ? 0 : 1,\n      indices: [0, text.length - 1]\n    }\n  }\n}\n\n// Token: ^file\n\nclass PrefixExactMatch extends BaseMatch {\n  constructor(pattern) {\n    super(pattern);\n  }\n  static get type() {\n    return 'prefix-exact'\n  }\n  static get multiRegex() {\n    return /^\\^\"(.*)\"$/\n  }\n  static get singleRegex() {\n    return /^\\^(.*)$/\n  }\n  search(text) {\n    const isMatch = text.startsWith(this.pattern);\n\n    return {\n      isMatch,\n      score: isMatch ? 0 : 1,\n      indices: [0, this.pattern.length - 1]\n    }\n  }\n}\n\n// Token: !^fire\n\nclass InversePrefixExactMatch extends BaseMatch {\n  constructor(pattern) {\n    super(pattern);\n  }\n  static get type() {\n    return 'inverse-prefix-exact'\n  }\n  static get multiRegex() {\n    return /^!\\^\"(.*)\"$/\n  }\n  static get singleRegex() {\n    return /^!\\^(.*)$/\n  }\n  search(text) {\n    const isMatch = !text.startsWith(this.pattern);\n\n    return {\n      isMatch,\n      score: isMatch ? 0 : 1,\n      indices: [0, text.length - 1]\n    }\n  }\n}\n\n// Token: .file$\n\nclass SuffixExactMatch extends BaseMatch {\n  constructor(pattern) {\n    super(pattern);\n  }\n  static get type() {\n    return 'suffix-exact'\n  }\n  static get multiRegex() {\n    return /^\"(.*)\"\\$$/\n  }\n  static get singleRegex() {\n    return /^(.*)\\$$/\n  }\n  search(text) {\n    const isMatch = text.endsWith(this.pattern);\n\n    return {\n      isMatch,\n      score: isMatch ? 0 : 1,\n      indices: [text.length - this.pattern.length, text.length - 1]\n    }\n  }\n}\n\n// Token: !.file$\n\nclass InverseSuffixExactMatch extends BaseMatch {\n  constructor(pattern) {\n    super(pattern);\n  }\n  static get type() {\n    return 'inverse-suffix-exact'\n  }\n  static get multiRegex() {\n    return /^!\"(.*)\"\\$$/\n  }\n  static get singleRegex() {\n    return /^!(.*)\\$$/\n  }\n  search(text) {\n    const isMatch = !text.endsWith(this.pattern);\n    return {\n      isMatch,\n      score: isMatch ? 0 : 1,\n      indices: [0, text.length - 1]\n    }\n  }\n}\n\nclass FuzzyMatch extends BaseMatch {\n  constructor(\n    pattern,\n    {\n      location = Config.location,\n      threshold = Config.threshold,\n      distance = Config.distance,\n      includeMatches = Config.includeMatches,\n      findAllMatches = Config.findAllMatches,\n      minMatchCharLength = Config.minMatchCharLength,\n      isCaseSensitive = Config.isCaseSensitive,\n      ignoreDiacritics = Config.ignoreDiacritics,\n      ignoreLocation = Config.ignoreLocation\n    } = {}\n  ) {\n    super(pattern);\n    this._bitapSearch = new BitapSearch(pattern, {\n      location,\n      threshold,\n      distance,\n      includeMatches,\n      findAllMatches,\n      minMatchCharLength,\n      isCaseSensitive,\n      ignoreDiacritics,\n      ignoreLocation\n    });\n  }\n  static get type() {\n    return 'fuzzy'\n  }\n  static get multiRegex() {\n    return /^\"(.*)\"$/\n  }\n  static get singleRegex() {\n    return /^(.*)$/\n  }\n  search(text) {\n    return this._bitapSearch.searchIn(text)\n  }\n}\n\n// Token: 'file\n\nclass IncludeMatch extends BaseMatch {\n  constructor(pattern) {\n    super(pattern);\n  }\n  static get type() {\n    return 'include'\n  }\n  static get multiRegex() {\n    return /^'\"(.*)\"$/\n  }\n  static get singleRegex() {\n    return /^'(.*)$/\n  }\n  search(text) {\n    let location = 0;\n    let index;\n\n    const indices = [];\n    const patternLen = this.pattern.length;\n\n    // Get all exact matches\n    while ((index = text.indexOf(this.pattern, location)) > -1) {\n      location = index + patternLen;\n      indices.push([index, location - 1]);\n    }\n\n    const isMatch = !!indices.length;\n\n    return {\n      isMatch,\n      score: isMatch ? 0 : 1,\n      indices\n    }\n  }\n}\n\n// ❗Order is important. DO NOT CHANGE.\nconst searchers = [\n  ExactMatch,\n  IncludeMatch,\n  PrefixExactMatch,\n  InversePrefixExactMatch,\n  InverseSuffixExactMatch,\n  SuffixExactMatch,\n  InverseExactMatch,\n  FuzzyMatch\n];\n\nconst searchersLen = searchers.length;\n\n// Regex to split by spaces, but keep anything in quotes together\nconst SPACE_RE = / +(?=(?:[^\\\"]*\\\"[^\\\"]*\\\")*[^\\\"]*$)/;\nconst OR_TOKEN = '|';\n\n// Return a 2D array representation of the query, for simpler parsing.\n// Example:\n// \"^core go$ | rb$ | py$ xy$\" => [[\"^core\", \"go$\"], [\"rb$\"], [\"py$\", \"xy$\"]]\nfunction parseQuery(pattern, options = {}) {\n  return pattern.split(OR_TOKEN).map((item) => {\n    let query = item\n      .trim()\n      .split(SPACE_RE)\n      .filter((item) => item && !!item.trim());\n\n    let results = [];\n    for (let i = 0, len = query.length; i < len; i += 1) {\n      const queryItem = query[i];\n\n      // 1. Handle multiple query match (i.e, once that are quoted, like `\"hello world\"`)\n      let found = false;\n      let idx = -1;\n      while (!found && ++idx < searchersLen) {\n        const searcher = searchers[idx];\n        let token = searcher.isMultiMatch(queryItem);\n        if (token) {\n          results.push(new searcher(token, options));\n          found = true;\n        }\n      }\n\n      if (found) {\n        continue\n      }\n\n      // 2. Handle single query matches (i.e, once that are *not* quoted)\n      idx = -1;\n      while (++idx < searchersLen) {\n        const searcher = searchers[idx];\n        let token = searcher.isSingleMatch(queryItem);\n        if (token) {\n          results.push(new searcher(token, options));\n          break\n        }\n      }\n    }\n\n    return results\n  })\n}\n\n// These extended matchers can return an array of matches, as opposed\n// to a singl match\nconst MultiMatchSet = new Set([FuzzyMatch.type, IncludeMatch.type]);\n\n/**\n * Command-like searching\n * ======================\n *\n * Given multiple search terms delimited by spaces.e.g. `^jscript .python$ ruby !java`,\n * search in a given text.\n *\n * Search syntax:\n *\n * | Token       | Match type                 | Description                            |\n * | ----------- | -------------------------- | -------------------------------------- |\n * | `jscript`   | fuzzy-match                | Items that fuzzy match `jscript`       |\n * | `=scheme`   | exact-match                | Items that are `scheme`                |\n * | `'python`   | include-match              | Items that include `python`            |\n * | `!ruby`     | inverse-exact-match        | Items that do not include `ruby`       |\n * | `^java`     | prefix-exact-match         | Items that start with `java`           |\n * | `!^earlang` | inverse-prefix-exact-match | Items that do not start with `earlang` |\n * | `.js$`      | suffix-exact-match         | Items that end with `.js`              |\n * | `!.go$`     | inverse-suffix-exact-match | Items that do not end with `.go`       |\n *\n * A single pipe character acts as an OR operator. For example, the following\n * query matches entries that start with `core` and end with either`go`, `rb`,\n * or`py`.\n *\n * ```\n * ^core go$ | rb$ | py$\n * ```\n */\nclass ExtendedSearch {\n  constructor(\n    pattern,\n    {\n      isCaseSensitive = Config.isCaseSensitive,\n      ignoreDiacritics = Config.ignoreDiacritics,\n      includeMatches = Config.includeMatches,\n      minMatchCharLength = Config.minMatchCharLength,\n      ignoreLocation = Config.ignoreLocation,\n      findAllMatches = Config.findAllMatches,\n      location = Config.location,\n      threshold = Config.threshold,\n      distance = Config.distance\n    } = {}\n  ) {\n    this.query = null;\n    this.options = {\n      isCaseSensitive,\n      ignoreDiacritics,\n      includeMatches,\n      minMatchCharLength,\n      findAllMatches,\n      ignoreLocation,\n      location,\n      threshold,\n      distance\n    };\n\n    pattern = isCaseSensitive ? pattern : pattern.toLowerCase();\n    pattern = ignoreDiacritics ? stripDiacritics(pattern) : pattern;\n    this.pattern = pattern;\n    this.query = parseQuery(this.pattern, this.options);\n  }\n\n  static condition(_, options) {\n    return options.useExtendedSearch\n  }\n\n  searchIn(text) {\n    const query = this.query;\n\n    if (!query) {\n      return {\n        isMatch: false,\n        score: 1\n      }\n    }\n\n    const { includeMatches, isCaseSensitive, ignoreDiacritics } = this.options;\n\n    text = isCaseSensitive ? text : text.toLowerCase();\n    text = ignoreDiacritics ? stripDiacritics(text) : text;\n\n    let numMatches = 0;\n    let allIndices = [];\n    let totalScore = 0;\n\n    // ORs\n    for (let i = 0, qLen = query.length; i < qLen; i += 1) {\n      const searchers = query[i];\n\n      // Reset indices\n      allIndices.length = 0;\n      numMatches = 0;\n\n      // ANDs\n      for (let j = 0, pLen = searchers.length; j < pLen; j += 1) {\n        const searcher = searchers[j];\n        const { isMatch, indices, score } = searcher.search(text);\n\n        if (isMatch) {\n          numMatches += 1;\n          totalScore += score;\n          if (includeMatches) {\n            const type = searcher.constructor.type;\n            if (MultiMatchSet.has(type)) {\n              allIndices = [...allIndices, ...indices];\n            } else {\n              allIndices.push(indices);\n            }\n          }\n        } else {\n          totalScore = 0;\n          numMatches = 0;\n          allIndices.length = 0;\n          break\n        }\n      }\n\n      // OR condition, so if TRUE, return\n      if (numMatches) {\n        let result = {\n          isMatch: true,\n          score: totalScore / numMatches\n        };\n\n        if (includeMatches) {\n          result.indices = allIndices;\n        }\n\n        return result\n      }\n    }\n\n    // Nothing was matched\n    return {\n      isMatch: false,\n      score: 1\n    }\n  }\n}\n\nconst registeredSearchers = [];\n\nfunction register(...args) {\n  registeredSearchers.push(...args);\n}\n\nfunction createSearcher(pattern, options) {\n  for (let i = 0, len = registeredSearchers.length; i < len; i += 1) {\n    let searcherClass = registeredSearchers[i];\n    if (searcherClass.condition(pattern, options)) {\n      return new searcherClass(pattern, options)\n    }\n  }\n\n  return new BitapSearch(pattern, options)\n}\n\nconst LogicalOperator = {\n  AND: '$and',\n  OR: '$or'\n};\n\nconst KeyType = {\n  PATH: '$path',\n  PATTERN: '$val'\n};\n\nconst isExpression = (query) =>\n  !!(query[LogicalOperator.AND] || query[LogicalOperator.OR]);\n\nconst isPath = (query) => !!query[KeyType.PATH];\n\nconst isLeaf = (query) =>\n  !isArray(query) && isObject(query) && !isExpression(query);\n\nconst convertToExplicit = (query) => ({\n  [LogicalOperator.AND]: Object.keys(query).map((key) => ({\n    [key]: query[key]\n  }))\n});\n\n// When `auto` is `true`, the parse function will infer and initialize and add\n// the appropriate `Searcher` instance\nfunction parse(query, options, { auto = true } = {}) {\n  const next = (query) => {\n    let keys = Object.keys(query);\n\n    const isQueryPath = isPath(query);\n\n    if (!isQueryPath && keys.length > 1 && !isExpression(query)) {\n      return next(convertToExplicit(query))\n    }\n\n    if (isLeaf(query)) {\n      const key = isQueryPath ? query[KeyType.PATH] : keys[0];\n\n      const pattern = isQueryPath ? query[KeyType.PATTERN] : query[key];\n\n      if (!isString(pattern)) {\n        throw new Error(LOGICAL_SEARCH_INVALID_QUERY_FOR_KEY(key))\n      }\n\n      const obj = {\n        keyId: createKeyId(key),\n        pattern\n      };\n\n      if (auto) {\n        obj.searcher = createSearcher(pattern, options);\n      }\n\n      return obj\n    }\n\n    let node = {\n      children: [],\n      operator: keys[0]\n    };\n\n    keys.forEach((key) => {\n      const value = query[key];\n\n      if (isArray(value)) {\n        value.forEach((item) => {\n          node.children.push(next(item));\n        });\n      }\n    });\n\n    return node\n  };\n\n  if (!isExpression(query)) {\n    query = convertToExplicit(query);\n  }\n\n  return next(query)\n}\n\n// Practical scoring function\nfunction computeScore(\n  results,\n  { ignoreFieldNorm = Config.ignoreFieldNorm }\n) {\n  results.forEach((result) => {\n    let totalScore = 1;\n\n    result.matches.forEach(({ key, norm, score }) => {\n      const weight = key ? key.weight : null;\n\n      totalScore *= Math.pow(\n        score === 0 && weight ? Number.EPSILON : score,\n        (weight || 1) * (ignoreFieldNorm ? 1 : norm)\n      );\n    });\n\n    result.score = totalScore;\n  });\n}\n\nfunction transformMatches(result, data) {\n  const matches = result.matches;\n  data.matches = [];\n\n  if (!isDefined(matches)) {\n    return\n  }\n\n  matches.forEach((match) => {\n    if (!isDefined(match.indices) || !match.indices.length) {\n      return\n    }\n\n    const { indices, value } = match;\n\n    let obj = {\n      indices,\n      value\n    };\n\n    if (match.key) {\n      obj.key = match.key.src;\n    }\n\n    if (match.idx > -1) {\n      obj.refIndex = match.idx;\n    }\n\n    data.matches.push(obj);\n  });\n}\n\nfunction transformScore(result, data) {\n  data.score = result.score;\n}\n\nfunction format(\n  results,\n  docs,\n  {\n    includeMatches = Config.includeMatches,\n    includeScore = Config.includeScore\n  } = {}\n) {\n  const transformers = [];\n\n  if (includeMatches) transformers.push(transformMatches);\n  if (includeScore) transformers.push(transformScore);\n\n  return results.map((result) => {\n    const { idx } = result;\n\n    const data = {\n      item: docs[idx],\n      refIndex: idx\n    };\n\n    if (transformers.length) {\n      transformers.forEach((transformer) => {\n        transformer(result, data);\n      });\n    }\n\n    return data\n  })\n}\n\nclass Fuse {\n  constructor(docs, options = {}, index) {\n    this.options = { ...Config, ...options };\n\n    if (\n      this.options.useExtendedSearch &&\n      !true\n    ) {\n      throw new Error(EXTENDED_SEARCH_UNAVAILABLE)\n    }\n\n    this._keyStore = new KeyStore(this.options.keys);\n\n    this.setCollection(docs, index);\n  }\n\n  setCollection(docs, index) {\n    this._docs = docs;\n\n    if (index && !(index instanceof FuseIndex)) {\n      throw new Error(INCORRECT_INDEX_TYPE)\n    }\n\n    this._myIndex =\n      index ||\n      createIndex(this.options.keys, this._docs, {\n        getFn: this.options.getFn,\n        fieldNormWeight: this.options.fieldNormWeight\n      });\n  }\n\n  add(doc) {\n    if (!isDefined(doc)) {\n      return\n    }\n\n    this._docs.push(doc);\n    this._myIndex.add(doc);\n  }\n\n  remove(predicate = (/* doc, idx */) => false) {\n    const results = [];\n\n    for (let i = 0, len = this._docs.length; i < len; i += 1) {\n      const doc = this._docs[i];\n      if (predicate(doc, i)) {\n        this.removeAt(i);\n        i -= 1;\n        len -= 1;\n\n        results.push(doc);\n      }\n    }\n\n    return results\n  }\n\n  removeAt(idx) {\n    this._docs.splice(idx, 1);\n    this._myIndex.removeAt(idx);\n  }\n\n  getIndex() {\n    return this._myIndex\n  }\n\n  search(query, { limit = -1 } = {}) {\n    const {\n      includeMatches,\n      includeScore,\n      shouldSort,\n      sortFn,\n      ignoreFieldNorm\n    } = this.options;\n\n    let results = isString(query)\n      ? isString(this._docs[0])\n        ? this._searchStringList(query)\n        : this._searchObjectList(query)\n      : this._searchLogical(query);\n\n    computeScore(results, { ignoreFieldNorm });\n\n    if (shouldSort) {\n      results.sort(sortFn);\n    }\n\n    if (isNumber(limit) && limit > -1) {\n      results = results.slice(0, limit);\n    }\n\n    return format(results, this._docs, {\n      includeMatches,\n      includeScore\n    })\n  }\n\n  _searchStringList(query) {\n    const searcher = createSearcher(query, this.options);\n    const { records } = this._myIndex;\n    const results = [];\n\n    // Iterate over every string in the index\n    records.forEach(({ v: text, i: idx, n: norm }) => {\n      if (!isDefined(text)) {\n        return\n      }\n\n      const { isMatch, score, indices } = searcher.searchIn(text);\n\n      if (isMatch) {\n        results.push({\n          item: text,\n          idx,\n          matches: [{ score, value: text, norm, indices }]\n        });\n      }\n    });\n\n    return results\n  }\n\n  _searchLogical(query) {\n\n    const expression = parse(query, this.options);\n\n    const evaluate = (node, item, idx) => {\n      if (!node.children) {\n        const { keyId, searcher } = node;\n\n        const matches = this._findMatches({\n          key: this._keyStore.get(keyId),\n          value: this._myIndex.getValueForItemAtKeyId(item, keyId),\n          searcher\n        });\n\n        if (matches && matches.length) {\n          return [\n            {\n              idx,\n              item,\n              matches\n            }\n          ]\n        }\n\n        return []\n      }\n\n      const res = [];\n      for (let i = 0, len = node.children.length; i < len; i += 1) {\n        const child = node.children[i];\n        const result = evaluate(child, item, idx);\n        if (result.length) {\n          res.push(...result);\n        } else if (node.operator === LogicalOperator.AND) {\n          return []\n        }\n      }\n      return res\n    };\n\n    const records = this._myIndex.records;\n    const resultMap = {};\n    const results = [];\n\n    records.forEach(({ $: item, i: idx }) => {\n      if (isDefined(item)) {\n        let expResults = evaluate(expression, item, idx);\n\n        if (expResults.length) {\n          // Dedupe when adding\n          if (!resultMap[idx]) {\n            resultMap[idx] = { idx, item, matches: [] };\n            results.push(resultMap[idx]);\n          }\n          expResults.forEach(({ matches }) => {\n            resultMap[idx].matches.push(...matches);\n          });\n        }\n      }\n    });\n\n    return results\n  }\n\n  _searchObjectList(query) {\n    const searcher = createSearcher(query, this.options);\n    const { keys, records } = this._myIndex;\n    const results = [];\n\n    // List is Array<Object>\n    records.forEach(({ $: item, i: idx }) => {\n      if (!isDefined(item)) {\n        return\n      }\n\n      let matches = [];\n\n      // Iterate over every key (i.e, path), and fetch the value at that key\n      keys.forEach((key, keyIndex) => {\n        matches.push(\n          ...this._findMatches({\n            key,\n            value: item[keyIndex],\n            searcher\n          })\n        );\n      });\n\n      if (matches.length) {\n        results.push({\n          idx,\n          item,\n          matches\n        });\n      }\n    });\n\n    return results\n  }\n  _findMatches({ key, value, searcher }) {\n    if (!isDefined(value)) {\n      return []\n    }\n\n    let matches = [];\n\n    if (isArray(value)) {\n      value.forEach(({ v: text, i: idx, n: norm }) => {\n        if (!isDefined(text)) {\n          return\n        }\n\n        const { isMatch, score, indices } = searcher.searchIn(text);\n\n        if (isMatch) {\n          matches.push({\n            score,\n            key,\n            value: text,\n            idx,\n            norm,\n            indices\n          });\n        }\n      });\n    } else {\n      const { v: text, n: norm } = value;\n\n      const { isMatch, score, indices } = searcher.searchIn(text);\n\n      if (isMatch) {\n        matches.push({ score, key, value: text, norm, indices });\n      }\n    }\n\n    return matches\n  }\n}\n\nFuse.version = '7.1.0';\nFuse.createIndex = createIndex;\nFuse.parseIndex = parseIndex;\nFuse.config = Config;\n\n{\n  Fuse.parseQuery = parse;\n}\n\n{\n  register(ExtendedSearch);\n}\n\nexport { Fuse as default };\n","import { useState, useCallback, useRef, useMemo } from \"react\";\nimport { useAtomValue } from \"jotai\";\nimport Fuse from \"fuse.js\";\n\nimport { autocompleteStateAtom } from \"src/atoms/globalSearch\";\nimport { amplitudeTrackEventAtom } from \"src/atoms/amplitude/amplitudeTrackEventAtom\";\nimport { SpiffyMetricsEventName } from \"src/contexts/amplitudeContext\";\nimport { SpiffyWidgets } from \"src/application/models/spiffyWidgets\";\n\nexport interface UseSearchInputProps {\n  initialSearchText?: string;\n  searchOrigin: SpiffyWidgets;\n  onSearchSubmit?: (query: string) => void;\n}\n\nexport interface UseSearchInputReturn {\n  // State\n  searchText: string;\n  hasText: boolean;\n  isFocused: boolean;\n  focusedIndex: number;\n  focusedOptionId: string | undefined;\n  autocompleteResults: string[];\n  shouldShowAutocomplete: boolean;\n  \n  // Handlers\n  handleSearchInputChange: (value: string) => void;\n  handleSearchInputFocus: () => void;\n  handleSearchInputBlur: () => void;\n  handleKeyDown: (event: React.KeyboardEvent<HTMLInputElement>) => void;\n  handleAutocompleteSelect: (suggestion: string, rankPosition?: number) => void;\n  handleSubmitSearch: () => void;\n  \n  // Utilities\n  setSearchText: (text: string) => void;\n  resetSearch: () => void;\n}\n\nexport const useSearchInput = ({\n  initialSearchText = \"\",\n  searchOrigin,\n  onSearchSubmit,\n}: UseSearchInputProps): UseSearchInputReturn => {\n  const track = useAtomValue(amplitudeTrackEventAtom);\n\n  // State\n  const [searchText, setSearchText] = useState(initialSearchText);\n  const [hasText, setHasText] = useState(initialSearchText.trim().length > 0);\n  const [isFocused, setIsFocused] = useState(false);\n  const [focusedIndex, setFocusedIndex] = useState(-1);\n  const [focusedOptionId, setFocusedOptionId] = useState<string | undefined>(\n    undefined\n  );\n  const autocompleteState = useAtomValue(autocompleteStateAtom);\n  const globalAutocompleteResults = autocompleteState.results;\n  const isAutocompleteEnabled = globalAutocompleteResults.length > 0;\n\n  // Refs\n  const isSelectingAutocomplete = useRef(false);\n\n  // Computed\n  const shouldShowAutocomplete = hasText && isFocused && isAutocompleteEnabled;\n\n  const autocompleteResultsLimit = 5;\n\n  // Configure Fuse.js for fuzzy search with custom scoring\n  const fuseOptions = useMemo(\n    () => ({\n      includeScore: true,\n      includeMatches: true,\n      // Fuzzy matching threshold (0.0 = perfect match, 1.0 = match anything)\n      threshold: 0.3,\n      // Location where match is expected (0 = beginning)\n      location: 0,\n      // Distance from location where match can be found - smaller for prefix preference\n      distance: 4,\n      // Minimum character length before fuzzy matching starts\n      minMatchCharLength: 1,\n      // Keys to search (we're searching strings directly, so no keys needed)\n      keys: [],\n      // Use extended search for exact matches and prefix matching\n      useExtendedSearch: false,\n      // Don't ignore location - this helps prioritize matches at the beginning\n      ignoreLocation: false,\n    }),\n    []\n  );\n\n  const fuse = useMemo(() => {\n    if (!globalAutocompleteResults.length) return null;\n    return new Fuse(globalAutocompleteResults, fuseOptions);\n  }, [globalAutocompleteResults, fuseOptions]);\n\n  // Autocomplete fuzzy search using Fuse.js\n  const autocompleteResults = useMemo(() => {\n    if (!isAutocompleteEnabled || !fuse) {\n      return [];\n    }\n\n    if (!searchText.trim()) {\n      return [];\n    }\n\n    const fuseResults = fuse.search(searchText);\n\n    // Use only Fuse.js scoring (results are already sorted by relevance)\n    return fuseResults\n      .slice(0, autocompleteResultsLimit)\n      .map((result) => result.item);\n  }, [isAutocompleteEnabled, fuse, searchText, autocompleteResultsLimit]);\n\n  // Handlers\n  const handleSearchInputChange = useCallback(\n    (newValue: string) => {\n      // If the user is changing text, they must be interacting with the input\n      // So we can safely assume they're focused\n      if (newValue !== searchText) {\n        setIsFocused(true);\n      }\n\n      if (newValue.length === 1) {\n        track?.({\n          eventName: SpiffyMetricsEventName.SearchInputStarted,\n          eventProps: {\n            searchOrigin,\n          },\n        });\n      }\n      setSearchText(newValue);\n      setHasText(newValue.trim().length > 0);\n      setFocusedIndex(-1);\n      setFocusedOptionId(undefined);\n    },\n    [searchOrigin, searchText, track]\n  );\n\n  const handleSearchInputFocus = useCallback(() => {\n    setIsFocused(true);\n  }, []);\n\n  const handleSearchInputBlur = useCallback(() => {\n    setTimeout(() => {\n      if (!isSelectingAutocomplete.current) {\n        setIsFocused(false);\n        setFocusedIndex(-1);\n        setFocusedOptionId(undefined);\n      }\n    }, 150);\n  }, []);\n\n  const handleSubmitSearch = useCallback(() => {\n    setIsFocused(false);\n    if (searchText.trim() && onSearchSubmit) {\n      track?.({\n        eventName: SpiffyMetricsEventName.SearchQuerySubmitted,\n        eventProps: {\n          searchOrigin,\n          queryText: searchText.trim(),\n        },\n        alsoSendToGoogleAnalytics: true,\n      });\n      onSearchSubmit(searchText.trim());\n    }\n  }, [searchText, onSearchSubmit, searchOrigin, track]);\n\n  const handleAutocompleteSelect = useCallback(\n    (suggestion: string, rankPosition?: number) => {\n      isSelectingAutocomplete.current = true;\n      setSearchText(suggestion);\n      setHasText(suggestion.trim().length > 0);\n      setIsFocused(false);\n\n      track?.({\n        eventName: SpiffyMetricsEventName.SearchInputStarted,\n        eventProps: {\n          searchOrigin,\n        },\n      });\n\n      if (onSearchSubmit) {\n        onSearchSubmit(suggestion.trim());\n      }\n\n      setTimeout(() => {\n        isSelectingAutocomplete.current = false;\n      }, 100);\n    },\n    [onSearchSubmit, searchOrigin, track]\n  );\n\n  const handleKeyDown = useCallback(\n    (event: React.KeyboardEvent<HTMLInputElement>) => {\n      if (event.key === \"ArrowDown\") {\n        event.preventDefault();\n        const newIndex = (focusedIndex + 1) % autocompleteResults.length;\n        setFocusedIndex(newIndex);\n        setFocusedOptionId(`option-${newIndex}`);\n      } else if (event.key === \"ArrowUp\") {\n        event.preventDefault();\n        const newIndex =\n          (focusedIndex - 1 + autocompleteResults.length) %\n          autocompleteResults.length;\n        setFocusedIndex(newIndex);\n        setFocusedOptionId(`option-${newIndex}`);\n      } else if (event.key === \"Enter\") {\n        if (focusedIndex === -1) {\n          event.preventDefault();\n          handleSubmitSearch();\n        } else {\n          event.preventDefault();\n          const suggestionText = autocompleteResults[focusedIndex];\n          handleAutocompleteSelect(suggestionText, focusedIndex);\n        }\n      } else if (event.key === \"Escape\") {\n        event.preventDefault();\n        setFocusedIndex(-1);\n        setFocusedOptionId(undefined);\n      }\n    },\n    [\n      autocompleteResults,\n      focusedIndex,\n      handleAutocompleteSelect,\n      handleSubmitSearch,\n    ]\n  );\n\n  const setSearchTextUtility = useCallback((text: string) => {\n    setSearchText(text);\n    setHasText(text.trim().length > 0);\n  }, []);\n\n  const resetSearch = useCallback(() => {\n    setSearchText(\"\");\n    setHasText(false);\n    setIsFocused(false);\n    setFocusedIndex(-1);\n    setFocusedOptionId(undefined);\n  }, []);\n\n  return {\n    // State\n    searchText,\n    hasText,\n    isFocused,\n    focusedIndex,\n    focusedOptionId,\n    autocompleteResults,\n    shouldShowAutocomplete,\n\n    // Handlers\n    handleSearchInputChange,\n    handleSearchInputFocus,\n    handleSearchInputBlur,\n    handleKeyDown,\n    handleAutocompleteSelect,\n    handleSubmitSearch,\n\n    // Utilities\n    setSearchText: setSearchTextUtility,\n    resetSearch,\n  };\n};","import { ProductRetrievalParams, ProductRetrievalResult } from \"./types\";\n\n// This will be set by the ProductRetrievalProvider when it initializes\nlet productRetrievalFunction:\n  | ((params: ProductRetrievalParams) => Promise<ProductRetrievalResult>)\n  | null = null;\n\nexport const setProductRetrievalFunction = (\n  fn: (params: ProductRetrievalParams) => Promise<ProductRetrievalResult>\n) => {\n  productRetrievalFunction = fn;\n};\n\nexport const getProductRetrievalFunction = () => {\n  if (!productRetrievalFunction) {\n    throw new Error(\n      \"Product retrieval function not initialized. Make sure ProductRetrievalProvider is mounted.\"\n    );\n  }\n  return productRetrievalFunction;\n};\n\nexport const clearProductRetrievalFunction = () => {\n  productRetrievalFunction = null;\n};","\nimport { atom } from \"jotai\";\nimport { getProductRetrievalFunction } from \"./productRetrievalAdapter\";\nimport { ProductRetrievalParams, ProductRetrievalResult } from \"./types\";\n\ninterface ProductRetrievalState {\n  data: ProductRetrievalResult | null;\n  loading: boolean;\n  error: string | null;\n  lastProductIds: string[] | null;\n}\n\nexport const productRetrievalAtom = atom<ProductRetrievalState>({\n  data: null,\n  loading: false,\n  error: null,\n  lastProductIds: null,\n});\n\nexport const performProductRetrievalAtom = atom(\n  null,\n  async (get, set, params: ProductRetrievalParams) => {\n    const currentState = get(productRetrievalAtom);\n\n    if (currentState.loading) {\n      return; // Prevent concurrent retrievals\n    }\n\n    set(productRetrievalAtom, {\n      data: null,\n      loading: true,\n      error: null,\n      lastProductIds: params.productIds,\n    });\n\n    try {\n      const retrieveProducts = getProductRetrievalFunction();\n      const result = await retrieveProducts(params);\n      set(productRetrievalAtom, {\n        data: result,\n        loading: false,\n        error: null,\n        lastProductIds: params.productIds,\n      });\n    } catch (error: unknown) {\n      const errorMessage =\n        error instanceof Error ? error.message : \"An unknown error occurred\";\n      set(productRetrievalAtom, {\n        data: null,\n        loading: false,\n        error: errorMessage,\n        lastProductIds: params.productIds,\n      });\n    }\n  }\n);\n\n// Get just the products array (already normalized by service)\nexport const retrievedProductsAtom = atom((get) => {\n  const retrievalData = get(productRetrievalAtom).data;\n  return retrievalData?.products || [];\n});\n\nexport const productRetrievalLoadingAtom = atom((get) => {\n  return get(productRetrievalAtom).loading;\n});\n\nexport const productRetrievalErrorAtom = atom((get) => {\n  return get(productRetrievalAtom).error;\n});","import { useAtomValue, useSetAtom } from \"jotai\";\nimport { useEffect } from \"react\";\nimport { useFeatureFlagService } from \"src/contexts/featureFlagServiceContext/featureFlagServiceContext\";\nimport {\n  retrievedProductsAtom,\n  performProductRetrievalAtom,\n  productRetrievalAtom,\n} from \"src/atoms/search/productRetrievalAPI\";\nimport { useNewOrgConfig } from \"../NewOrgConfig\";\nimport { SearchResponseProductAttributes } from \"src/application/models/api/response\";\nimport { FeatureGates } from \"src/application/models\";\n\n/**\n * A hook to get recommended products.\n * It checks the `IsRecommendedProductsEnabled` feature gate and triggers fetching if needed.\n * If the gate is enabled, it returns the list of recommended products.\n * If the gate is disabled, it returns an empty array.\n *\n * @returns The list of recommended products or an empty array.\n */\nexport const useRecommendedProducts =\n  (): SearchResponseProductAttributes[\"attributes\"][] => {\n    const { featureFlagService } = useFeatureFlagService();\n    const isEnabled = featureFlagService?.isFeatureGateEnabled(\n      FeatureGates.IsRecommendedProductsEnabled\n    );\n    const recommendedProducts = useAtomValue(retrievedProductsAtom);\n    const performProductRetrieval = useSetAtom(performProductRetrievalAtom);\n    const { lastProductIds } = useAtomValue(productRetrievalAtom);\n    const orgUIConfig = useNewOrgConfig();\n\n    useEffect(() => {\n      const recommendedProductIds = orgUIConfig.frontendConfig?.uiConfigs?.searchConfig\n        ? orgUIConfig.frontendConfig?.uiConfigs?.searchConfig.recommendedProducts\n        : [];\n      // Fetch only if the feature is enabled, there are IDs to fetch, and we haven't already fetched them.\n      if (\n        isEnabled &&\n        recommendedProductIds &&\n        recommendedProductIds.length > 0 &&\n        !lastProductIds\n      ) {\n        performProductRetrieval({ productIds: recommendedProductIds });\n      }\n    }, [isEnabled, performProductRetrieval, lastProductIds, orgUIConfig]);\n\n    return isEnabled ? recommendedProducts : [];\n  };","import { useAtom, useAtomValue, useSetAtom } from \"jotai\";\nimport { useCallback, useEffect, useMemo, useRef } from \"react\";\n\nimport {\n  addSearchFilterAtom,\n  clearSearchFiltersAtom,\n  createFilterOption,\n  filteredSearchProductsAtom,\n  performSearchAtom,\n  removeSearchFilterAtom,\n  searchAtom,\n  searchFiltersAtom,\n  searchParamsAtom,\n  searchProductSortingAtom,\n  searchSelectedFiltersAtom,\n  SelectedFilterOption,\n} from \"src/atoms/search\";\nimport {\n  isFilterOpenAtom,\n} from \"src/atoms/globalSearch/globalSearch\";\nimport { formatFilterDisplayName } from \"src/atoms/search/utils\";\nimport { ProductSorting } from \"src/atoms/search/types\";\nimport {\n  SpiffyMetricsEventName,\n  useAmplitude,\n} from \"src/contexts/amplitudeContext/amplitudeContext\";\nimport { SpiffyWidgets } from \"src/application/models/spiffyWidgets\";\nimport { ProductCardConfig } from \"src/contexts/types\";\nimport { SearchResult } from \"src/application/models/api/search\";\nimport { SearchResponseProduct } from \"@spiffy-ai/commerce-api-client\";\nimport {\n  SearchFilterDatum,\n  SelectFilterItem,\n} from \"src/types/search-filter-types\";\nimport { getSearchResultsState, SearchResultsState } from \"../utils\";\nimport { orgShortNameAtom } from \"src/atoms/envive/enviveConfig\";\nimport { useTrackComponentVisibleEvent } from \"../TrackComponentVisibleEvent\";\nimport { amplitudeTrackEventAtom } from \"src/atoms/amplitude/amplitudeTrackEventAtom\";\nimport { newOrgConfigAtom } from \"src/atoms/org\";\nimport { retrievedProductsAtom } from \"src/atoms/search/productRetrievalAPI\";\nimport { useSearchInput } from \"./useSearchInput\";\nimport { SearchResponseProductAttributes } from \"src/application/models\";\nimport { useNewOrgConfig } from \"../NewOrgConfig\";\nimport { useRecommendedProducts } from \"./useRecommendedProducts\";\n\nexport interface SearchResultsHocProps {\n  // Data\n  searchData: SearchResult | null;\n  searchResponseId: string;\n  merchantShortName: string;\n  productCardConfig: ProductCardConfig;\n  productList: SearchResponseProduct[];\n  autocompleteResults: string[];\n  searchFilters: SearchFilterDatum[];\n  availableDynamicFilters: { name: string; displayName: string }[];\n  selectedFilterOptions: SelectedFilterOption[];\n  recommendedProducts: SearchResponseProductAttributes['attributes'][];\n\n  // State\n  searchText: string;\n  query: string;\n  searchResultsState: SearchResultsState;\n  isLoadingSearch: boolean;\n  isFilterOpen: boolean;\n  shouldShowAutocomplete: boolean;\n  focusedIndex: number;\n  focusedOptionId: string | undefined;\n\n  // UI\n  filterButtonText: string;\n\n  // Event Handlers\n  onSearchInputChange: (value: string) => void;\n  onSubmitSearch: () => void;\n  onAutocompleteSelect: (suggestion: string) => void;\n  onKeyDown: (event: React.KeyboardEvent<HTMLInputElement>) => void;\n  onSearchInputFocus: () => void;\n  onSearchInputBlur: () => void;\n  onToggleDynamicFilter: ({ filter, dynamicFilterDisplayName }: { filter: string; dynamicFilterDisplayName: string }) => void;\n  onSelectFilterItem: SelectFilterItem;\n  onRemoveFilter: (filter: SelectedFilterOption) => void;\n  onClearAllFilters: () => void;\n  setIsFilterOpen: (isFilterOpen: boolean) => void;\n\n  // Refs\n  searchResultsRef: React.RefObject<HTMLDivElement>;\n}\n\nexport const useSearch = (): SearchResultsHocProps => {\n  // Atoms\n  const config = useNewOrgConfig();\n  const orgShortName = useAtomValue(orgShortNameAtom);\n  const { data: searchData, loading: isLoadingSearch } =\n    useAtomValue(searchAtom);\n  const productList = useAtomValue(filteredSearchProductsAtom);\n  const performSearch = useSetAtom(performSearchAtom);\n  const [{ query }] = useAtom(searchParamsAtom);\n  const [isFilterOpen, setIsFilterOpen] = useAtom(isFilterOpenAtom);\n  const [selectedFilterOptions] = useAtom(searchSelectedFiltersAtom);\n  const addFilter = useSetAtom(addSearchFilterAtom);\n  const removeFilter = useSetAtom(removeSearchFilterAtom);\n  const [productSorting, setProductSorting] = useAtom(searchProductSortingAtom);\n  const clearFilters = useSetAtom(clearSearchFiltersAtom);\n  const searchFilters = useAtomValue(searchFiltersAtom);\n  const recommendedProducts = useRecommendedProducts();\n\n  // Refs\n  const searchResultsRef = useRef<HTMLDivElement>(null);\n  \n  // Utilities\n  const scrollToTop = useCallback(() => {\n    const container = searchResultsRef.current;\n    if (container) {\n      container.scrollTo({ top: 0, behavior: 'smooth' });\n    } else {\n      window.scrollTo({ top: 0, behavior: 'smooth' });\n    }\n  }, []);\n  \n  // Search Input Hook\n  const searchInput = useSearchInput({\n    initialSearchText: query || '',\n    searchOrigin: SpiffyWidgets.SearchResults,\n    onSearchSubmit: (searchQuery) => {\n      performSearch({ query: searchQuery });\n    },\n  });\n  \n  const {\n    searchText,\n    focusedIndex,\n    focusedOptionId,\n    autocompleteResults,\n    shouldShowAutocomplete,\n    handleSearchInputChange,\n    handleSearchInputFocus,\n    handleSearchInputBlur,\n    handleKeyDown,\n    handleAutocompleteSelect,\n    handleSubmitSearch,\n    setSearchText,\n  } = searchInput;\n  const searchResultsState = getSearchResultsState(isLoadingSearch, searchData);\n\n  const dynamicFilters = searchData?.filters || [];\n\n  // Provide fallback values when orgUIConfig is not yet available\n  const safeProductCardConfig = config?.frontendConfig?.uiConfigs\n    ?.productCardConfig || {\n    variant: \"minimal\",\n    hoverVariant: \"none\",\n    layoutVariant: \"square\",\n  };\n  const safeMerchantShortName = orgShortName || \"\";\n\n  const availableDynamicFilters = useMemo(() => {\n    return dynamicFilters\n      .filter(\n        (dynamicFilterName) =>\n          !selectedFilterOptions.some((option) => option.id === `dynamic:${dynamicFilterName}`),\n      )\n      .map((dynamicFilterName) => ({\n        name: dynamicFilterName,\n        displayName: formatFilterDisplayName(dynamicFilterName),\n      }));\n  }, [dynamicFilters, selectedFilterOptions]);\n\n  const filters = useMemo(() => {\n    const sortOptions = [\n      {\n        filterItemId: String(ProductSorting.FEATURED),\n        displayName: 'Relevance',\n        productCount: 0,\n        isSelected: productSorting === ProductSorting.FEATURED,\n      },\n      {\n        filterItemId: String(ProductSorting.PRICE_ASC),\n        displayName: 'Price: Low to High',\n        productCount: 0,\n        isSelected: productSorting === ProductSorting.PRICE_ASC,\n      },\n      {\n        filterItemId: String(ProductSorting.PRICE_DESC),\n        displayName: 'Price: High to Low',\n        productCount: 0,\n        isSelected: productSorting === ProductSorting.PRICE_DESC,\n      },\n    ];\n\n    return [{ filterId: 'sort', displayName: 'SORT', items: sortOptions }, ...searchFilters] as SearchFilterDatum[];\n  }, [productSorting, searchFilters]);\n\n  const filterButtonText = useMemo(() => {\n    const selectedCount = filters.reduce((acc: number, filter) => {\n      if (filter.filterId === 'sort') {\n        return acc;\n      }\n      return acc + filter.items.filter((item) => item.isSelected).length;\n    }, 0);\n    if (selectedCount === 0) {\n      return 'Filter & Sort';\n    }\n    return `Filter & Sort (${selectedCount})`;\n  }, [filters]);\n\n  // Callbacks\n  const { trackEvent } = useAmplitude();\n\n\n  const handleToggleDynamicFilter = useCallback(\n    ({ filter, dynamicFilterDisplayName }: { filter: string; dynamicFilterDisplayName: string }) => {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.SearchFilterClicked,\n        eventProps: {\n          filterType: 'Dynamic',\n          filterValue: filter,\n          queryText: searchText,\n        },\n      });\n      addFilter(createFilterOption('dynamic', filter, dynamicFilterDisplayName));\n      scrollToTop();\n    },\n    [addFilter, searchText, scrollToTop],\n  );\n\n  const handleRemoveFilter = useCallback(\n    (filter: SelectedFilterOption) => {\n      removeFilter(filter.id);\n      scrollToTop();\n    },\n    [removeFilter, scrollToTop],\n  );\n\n\n  const handleSelectFilterItem: SelectFilterItem = useCallback(\n    ({\n      filterId,\n      filterItemId,\n      isSelected,\n      displayName,\n    }: {\n      filterId: string;\n      filterItemId: string;\n      isSelected: boolean;\n      displayName: string;\n    }) => {\n      if (filterId === 'sort') {\n        const newSort = filterItemId as ProductSorting;\n        trackEvent({\n          eventName: SpiffyMetricsEventName.SearchSortClicked,\n          eventProps: {\n            sortType: newSort,\n            queryText: searchText,\n          },\n        });\n        setProductSorting(newSort);\n        scrollToTop();\n      } else if (!isSelected) {\n        removeFilter(`${filterId}:${filterItemId}`);\n        scrollToTop();\n      } else {\n        trackEvent({\n          eventName: SpiffyMetricsEventName.SearchFilterClicked,\n          eventProps: {\n            filterType: 'Static',\n            filterCategory: filterId,\n            filterValue: filterItemId,\n            queryText: searchText,\n          },\n        });\n        addFilter(createFilterOption(filterId, filterItemId, displayName));\n        scrollToTop();\n      }\n    },\n    [addFilter, removeFilter, setProductSorting, searchText, scrollToTop],\n  );\n\n\n  const handleClearAllFilters = useCallback(() => {\n    setProductSorting(ProductSorting.FEATURED);\n    clearFilters();\n    scrollToTop();\n  }, [setProductSorting, clearFilters, scrollToTop]);\n\n  // Side Effects\n  useTrackComponentVisibleEvent(\n    SpiffyWidgets.SearchResults,\n    searchResultsRef,\n    {},\n    SpiffyMetricsEventName.SearchComponentVisible,\n  );\n\n  useEffect(() => {\n    if (searchResultsState === SearchResultsState.Results || searchResultsState === SearchResultsState.NoResults) {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.SearchResultsViewed,\n        eventProps: {\n          queryText: searchText,\n          resultsCount: productList.length,\n        },\n      });\n    }\n  }, [productList.length, searchResultsState]);\n\n  // This is the single source of truth for the search text triggering a search\n  useEffect(() => {\n    if (query && query !== searchText) {\n      const esq = new URLSearchParams(window.location.search).get('esq');\n      setSearchText(query);\n      performSearch({ query: esq ?? query });\n    }\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [performSearch, query, setSearchText]);\n\n  return {\n    searchData,\n    searchResponseId: searchData?.searchResponseId ?? '',\n    merchantShortName: safeMerchantShortName,\n    productCardConfig: safeProductCardConfig,\n    productList,\n    recommendedProducts,\n    autocompleteResults,\n    searchFilters: filters,\n    availableDynamicFilters,\n    selectedFilterOptions,\n    searchText,\n    query: query || '',\n    searchResultsState,\n    isLoadingSearch,\n    isFilterOpen,\n    shouldShowAutocomplete,\n    focusedIndex,\n    focusedOptionId,\n    filterButtonText,\n    onSearchInputChange: handleSearchInputChange,\n    onSubmitSearch: handleSubmitSearch,\n    onAutocompleteSelect: handleAutocompleteSelect,\n    onKeyDown: handleKeyDown,\n    onSearchInputFocus: handleSearchInputFocus,\n    onSearchInputBlur: handleSearchInputBlur,\n    onToggleDynamicFilter: handleToggleDynamicFilter,\n    onSelectFilterItem: handleSelectFilterItem,\n    onRemoveFilter: handleRemoveFilter,\n    onClearAllFilters: handleClearAllFilters,\n    setIsFilterOpen,\n    searchResultsRef,\n  };\n};"],"x_google_ignoreList":[0],"mappings":"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"}