@envive-ai/react-hooks 0.1.5 → 0.1.6

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 (491) hide show
  1. package/dist/amplitudeContext-CHzas7Cf.d.cts +52 -0
  2. package/dist/amplitudeContext-D-7fmVh1.cjs +356 -0
  3. package/dist/amplitudeContext-hY3caPC6.js +322 -0
  4. package/dist/amplitudeContext-tJ4y-SzX.d.ts +52 -0
  5. package/dist/api-DjeZXxl_.js +207 -0
  6. package/dist/api-_JaUnIUj.cjs +292 -0
  7. package/dist/app-CjsQ2_n-.js +132 -0
  8. package/dist/app-CnKRZ9RW.cjs +178 -0
  9. package/dist/application/models/graphql/index.cjs +3 -12
  10. package/dist/application/models/graphql/index.d.cts +1 -1
  11. package/dist/application/models/graphql/index.d.ts +1 -1
  12. package/dist/application/models/graphql/index.js +1 -10
  13. package/dist/application/models/guards/api/index.cjs +1 -2
  14. package/dist/application/models/guards/api/index.d.cts +2 -2
  15. package/dist/application/models/guards/api/index.d.ts +2 -2
  16. package/dist/application/models/guards/api/index.js +1 -2
  17. package/dist/application/models/index.cjs +48 -54
  18. package/dist/application/models/index.d.cts +7 -9
  19. package/dist/application/models/index.d.ts +7 -9
  20. package/dist/application/models/index.js +7 -11
  21. package/dist/application/models/utilityTypes/index.cjs +1 -1
  22. package/dist/application/models/utilityTypes/index.d.cts +1 -1
  23. package/dist/application/models/utilityTypes/index.d.ts +1 -1
  24. package/dist/application/models/utilityTypes/index.js +1 -1
  25. package/dist/application/models/variantInfo/index.cjs +1 -1
  26. package/dist/application/models/variantInfo/index.d.cts +1 -1
  27. package/dist/application/models/variantInfo/index.d.ts +1 -1
  28. package/dist/application/models/variantInfo/index.js +1 -1
  29. package/dist/application/utils/index.cjs +40 -37
  30. package/dist/application/utils/index.d.cts +18 -14
  31. package/dist/application/utils/index.d.ts +19 -15
  32. package/dist/application/utils/index.js +13 -10
  33. package/dist/{atomStore-JwGb7pcy.cjs → atomStore-B4jIaDPd.cjs} +1 -1
  34. package/dist/{atomStore-B1cgmbP0.js → atomStore-D8pjE1vL.js} +1 -1
  35. package/dist/atoms/app/index.cjs +14 -16
  36. package/dist/atoms/app/index.d.cts +20 -19
  37. package/dist/atoms/app/index.d.ts +20 -19
  38. package/dist/atoms/app/index.js +10 -11
  39. package/dist/atoms/atomStore/index.cjs +2 -1
  40. package/dist/atoms/atomStore/index.d.cts +24 -2
  41. package/dist/atoms/atomStore/index.d.ts +24 -2
  42. package/dist/atoms/atomStore/index.js +2 -1
  43. package/dist/atoms/chat/index.cjs +44 -40
  44. package/dist/atoms/chat/index.d.cts +35 -36
  45. package/dist/atoms/chat/index.d.ts +35 -36
  46. package/dist/atoms/chat/index.js +14 -10
  47. package/dist/atoms/globalSearch/index.cjs +1 -1
  48. package/dist/atoms/globalSearch/index.d.cts +5 -5
  49. package/dist/atoms/globalSearch/index.d.ts +6 -6
  50. package/dist/atoms/globalSearch/index.js +1 -1
  51. package/dist/atoms/org/index.cjs +26 -38
  52. package/dist/atoms/org/index.d.cts +35 -48
  53. package/dist/atoms/org/index.d.ts +35 -48
  54. package/dist/atoms/org/index.js +11 -11
  55. package/dist/atoms/search/index.cjs +47 -41
  56. package/dist/atoms/search/index.d.cts +7 -9
  57. package/dist/atoms/search/index.d.ts +7 -9
  58. package/dist/atoms/search/index.js +16 -10
  59. package/dist/bandolier-DYEz4-8C.js +1229 -0
  60. package/dist/bandolier-o_-xrbOV.cjs +1229 -0
  61. package/dist/carpe-CsG5jCcl.cjs +607 -0
  62. package/dist/carpe-DONk00bZ.js +605 -0
  63. package/dist/cdnUtils-32v8bDX9.cjs +16 -0
  64. package/dist/cdnUtils-E4tLBt2g.js +11 -0
  65. package/dist/{chat-_NzpO_Hn.cjs → chat-BubCW1as.cjs} +1 -1
  66. package/dist/chat-CO7cRkaq.cjs +506 -0
  67. package/dist/{chat-CHmD79E1.js → chat-CV6MXeID.js} +1 -1
  68. package/dist/chat-hcRc4RRd.js +285 -0
  69. package/dist/{components-Dl-TMD9k.js → components-B4T3Uzth.js} +1 -1
  70. package/dist/{components-D0XDRcaN.cjs → components-Cw9WjA6g.cjs} +1 -1
  71. package/dist/config/locators/components/chat/index.cjs +1 -1
  72. package/dist/config/locators/components/chat/index.d.cts +1 -1
  73. package/dist/config/locators/components/chat/index.d.ts +1 -1
  74. package/dist/config/locators/components/chat/index.js +1 -1
  75. package/dist/config/locators/components/chat/variants/index.d.cts +1 -1
  76. package/dist/config/locators/components/chat/variants/index.d.ts +1 -1
  77. package/dist/config/locators/components/index.cjs +1 -1
  78. package/dist/config/locators/components/index.d.cts +1 -1
  79. package/dist/config/locators/components/index.d.ts +1 -1
  80. package/dist/config/locators/components/index.js +1 -1
  81. package/dist/config/locators/components/search/index.d.ts +1 -1
  82. package/dist/config/locators/index.cjs +4 -4
  83. package/dist/config/locators/index.d.cts +2 -2
  84. package/dist/config/locators/index.d.ts +2 -2
  85. package/dist/config/locators/index.js +4 -4
  86. package/dist/contexts/index.cjs +45 -14
  87. package/dist/contexts/index.d.cts +10 -10
  88. package/dist/contexts/index.d.ts +10 -10
  89. package/dist/contexts/index.js +22 -11
  90. package/dist/contexts-B4ihTBsV.cjs +2980 -0
  91. package/dist/contexts-DGlr7M1o.js +2596 -0
  92. package/dist/coterie-CKB93qfz.cjs +239 -0
  93. package/dist/{coterie--MGvWeVh.cjs → coterie-DHcj2fRE.js} +64 -65
  94. package/dist/{custservice-types-C8-9vKWb.cjs → custservice-types-Dy0kc9TW.cjs} +1 -1
  95. package/dist/{custservice-types-CamCiXjq.js → custservice-types-mnIunabv.js} +1 -1
  96. package/dist/default-BnB4p0Se.cjs +234 -0
  97. package/dist/default-D6f5Dyt7.js +199 -0
  98. package/dist/default-ZKkE5zT4.cjs +4 -0
  99. package/dist/default-legGZJwI.js +4 -0
  100. package/dist/{divIds-CjceLRD9.cjs → divIds-2dJlU3z8.cjs} +1 -15
  101. package/dist/{divIds-0Vj9g-fM.js → divIds-BUrN2vY-.js} +2 -4
  102. package/dist/dreamlandBaby-BiRYYFav.js +346 -0
  103. package/dist/dreamlandBaby-zuvmfzfD.cjs +346 -0
  104. package/dist/{embedded-_cLpd6FK.js → embedded-BJLWIriJ.js} +1 -1
  105. package/dist/{embedded-B5Wi9g0T.cjs → embedded-Dl79TJLC.cjs} +1 -1
  106. package/dist/{entrypoints-CNUvSgnk.js → entrypoints-Bs3pn6EV.js} +1 -1
  107. package/dist/{entrypoints-DM9Sm18O.cjs → entrypoints-Dqi5pzWH.cjs} +1 -1
  108. package/dist/enviveConfig-BccuzS2a.cjs +240 -0
  109. package/dist/enviveConfig-CzhTz8Aa.js +152 -0
  110. package/dist/events/index.cjs +1 -1
  111. package/dist/events/index.js +1 -1
  112. package/dist/{events-ClCDFK7t.js → events-BrLpaNxh.js} +1 -1
  113. package/dist/{events-Da7gpmGv.cjs → events-DwT6cEwI.cjs} +1 -1
  114. package/dist/fiveCbd-DwTTwC2j.cjs +613 -0
  115. package/dist/fiveCbd-Dz2NouOF.js +613 -0
  116. package/dist/forLoveAndLemons-DqSdyD6S.js +665 -0
  117. package/dist/forLoveAndLemons-Ud6mPrV3.cjs +667 -0
  118. package/dist/{globalSearch-b0wC7ZEO.js → globalSearch-B6THR7Jx.js} +1 -1
  119. package/dist/{globalSearch-BTeZxvk1.cjs → globalSearch-BFvEg53C.cjs} +1 -1
  120. package/dist/graphql-BNW60InJ.cjs +128 -0
  121. package/dist/graphql-CjB8wHzm.js +74 -0
  122. package/dist/graphqlConfig-CYt6tWII.js +30 -0
  123. package/dist/graphqlConfig-DicykC-o.cjs +61 -0
  124. package/dist/greenpan-B_EbPkSP.js +397 -0
  125. package/dist/greenpan-kPE4fJgd.cjs +397 -0
  126. package/dist/grooveLife-Ckuir319.js +342 -0
  127. package/dist/grooveLife-DKSEQp1F.cjs +342 -0
  128. package/dist/homegrownCannabis-BVPa2pqe.js +410 -0
  129. package/dist/homegrownCannabis-BwIAkxuA.cjs +410 -0
  130. package/dist/hooks/index.cjs +39 -13
  131. package/dist/hooks/index.d.cts +127 -45
  132. package/dist/hooks/index.d.ts +124 -42
  133. package/dist/hooks/index.js +22 -11
  134. package/dist/{index-D46Rd0io.d.cts → index-B0NW9KTj.d.cts} +1 -1
  135. package/dist/{index-BdNKc2ix.d.cts → index-BDPWBU3h.d.cts} +1 -1
  136. package/dist/index-BMPtlgac.d.ts +191 -0
  137. package/dist/{index-BTK5uzs6.d.cts → index-Bmub8e38.d.cts} +1 -1
  138. package/dist/{index-BrXuc_Ck.d.cts → index-Byb45OPg.d.cts} +1 -1
  139. package/dist/{index-CSIOQD-A.d.ts → index-C6KdON7d.d.ts} +1 -1
  140. package/dist/index-C7pGDl1A.d.ts +1609 -0
  141. package/dist/{index-K2kNsOTw.d.cts → index-C9lgsCWp.d.cts} +32 -32
  142. package/dist/{index-BzgkfbNO.d.cts → index-CESxqFso.d.cts} +1 -1
  143. package/dist/{index-CbJZOEU4.d.ts → index-Cc-wBCn6.d.ts} +32 -32
  144. package/dist/{index-Cx9e-fRi.d.ts → index-CcIG01PJ.d.ts} +2 -2
  145. package/dist/{index-ClJ0nMsR.d.cts → index-CiWEYzXl.d.cts} +2 -2
  146. package/dist/{index-CKKkTsms.d.ts → index-Cqg6ltII.d.ts} +1 -1
  147. package/dist/{index-Cr2y08f1.d.ts → index-DOii3C6b.d.ts} +1 -1
  148. package/dist/index-DTu2X-YS.d.cts +1609 -0
  149. package/dist/index-OEifcBsm.d.cts +191 -0
  150. package/dist/{index-Dxpscrvz.d.ts → index-PGF8BvxQ.d.ts} +1 -1
  151. package/dist/{index-D2VaMPA3.d.ts → index-ylUiJvnr.d.ts} +1 -1
  152. package/dist/interceptors/index.cjs +2 -12
  153. package/dist/interceptors/index.d.cts +8 -13
  154. package/dist/interceptors/index.d.ts +8 -13
  155. package/dist/interceptors/index.js +2 -11
  156. package/dist/jackArcher-B39OEIj6.cjs +727 -0
  157. package/dist/{jackArcher-sO9EbgrZ.js → jackArcher-BwkWGybC.js} +24 -25
  158. package/dist/jordanCraig-CxRH_jLi.js +1786 -0
  159. package/dist/jordanCraig-RwmWw-jM.cjs +1786 -0
  160. package/dist/kindredBravely-D78VwL20.cjs +490 -0
  161. package/dist/kindredBravely-DQDpAzdl.js +490 -0
  162. package/dist/kutFromTheKloth-TcXQkIti.js +369 -0
  163. package/dist/kutFromTheKloth-dXRrNti0.cjs +369 -0
  164. package/dist/larryAndSerges-Bh5XEXsZ.js +262 -0
  165. package/dist/larryAndSerges-COdBzOzu.cjs +262 -0
  166. package/dist/leapsAndRebounds-BbXnqkl5.cjs +360 -0
  167. package/dist/leapsAndRebounds-mjCaH1k-.js +360 -0
  168. package/dist/longevityrx-DdV3v26F.cjs +320 -0
  169. package/dist/longevityrx-DfPDfvAt.js +320 -0
  170. package/dist/lookOptic-DgXFGBPP.cjs +282 -0
  171. package/dist/{lookOptic-Jwf7EAU8.js → lookOptic-FgVW1afF.js} +19 -20
  172. package/dist/mantraBrand-C5gVm3rk.cjs +750 -0
  173. package/dist/{mantraBrand-DoaVj837.js → mantraBrand-uV8HCDR8.js} +22 -23
  174. package/dist/medterra-BtDpr1Hw.cjs +583 -0
  175. package/dist/medterra-DgzE7-mj.js +583 -0
  176. package/dist/modells-CJjsxOIR.js +484 -0
  177. package/dist/modells-Ck5KbRFj.cjs +484 -0
  178. package/dist/models-BkXg5eIP.cjs +1534 -0
  179. package/dist/models-UZ6HszfZ.js +1281 -0
  180. package/dist/{nodeSelector-DybpVr-i.d.ts → nodeSelector-BxrS2S_k.d.ts} +1 -1
  181. package/dist/{nodeSelector-vKB44CDB.d.cts → nodeSelector-Dm4S-10n.d.cts} +1 -1
  182. package/dist/org-C2RrBVQR.cjs +81 -0
  183. package/dist/org-xMytX--e.js +38 -0
  184. package/dist/orgAnalyticsConfig-BJ2A1RZ-.cjs +39 -0
  185. package/dist/orgAnalyticsConfig-ClrFy2qH.js +14 -0
  186. package/dist/pressedFloral-Bteoboms.js +661 -0
  187. package/dist/pressedFloral-CexgV-xy.cjs +661 -0
  188. package/dist/search-BMOAmrmY.js +459 -0
  189. package/dist/search-Df0Ifneg.cjs +672 -0
  190. package/dist/{search-filter-types-9rTb3jMj.d.cts → search-filter-types-DosbseF3.d.cts} +1 -1
  191. package/dist/{search-filter-types-C-zZSpGo.d.ts → search-filter-types-fZf91Pdw.d.ts} +1 -1
  192. package/dist/skinPerfection-BGzq9lq-.cjs +334 -0
  193. package/dist/skinPerfection-BlvcEcUE.js +334 -0
  194. package/dist/snapSupplements-CbbGzAgO.cjs +285 -0
  195. package/dist/snapSupplements-kcsPAOm9.js +285 -0
  196. package/dist/{socialProofClasses-ky69yppk.cjs → socialProofClasses-Db8gzsfi.cjs} +1 -1
  197. package/dist/{socialProofClasses-DdzG1tZy.js → socialProofClasses-kwDvwLOZ.js} +1 -1
  198. package/dist/spanx-B4WFA_rI.js +661 -0
  199. package/dist/spanx-BWoE4F8b.cjs +663 -0
  200. package/dist/spanxStaging-BOrOjhXn.js +845 -0
  201. package/dist/spanxStaging-BfdfIug4.cjs +848 -0
  202. package/dist/{suggestionBarV2-types-Penx3Y67.js → suggestionBarV2-types-IMMOmCir.js} +1 -1
  203. package/dist/{suggestionBarV2-types-B3lwrENK.cjs → suggestionBarV2-types-nnGNgFvR.cjs} +1 -1
  204. package/dist/supergoop-22dd5_BS.js +323 -0
  205. package/dist/supergoop-B-a4cku2.cjs +323 -0
  206. package/dist/{test-types-CXVJxTeu.d.ts → test-types-C9b_OdfO.d.ts} +1 -1
  207. package/dist/{test-types-CuOq25VT.d.cts → test-types-CgVJtwUr.d.cts} +1 -1
  208. package/dist/types/index.cjs +2 -2
  209. package/dist/types/index.d.cts +2 -2
  210. package/dist/types/index.d.ts +2 -2
  211. package/dist/types/index.js +2 -2
  212. package/dist/types--pr1GQQx.js +154 -0
  213. package/dist/types-BVSyY3Hk.cjs +196 -0
  214. package/dist/{uniqueVintage-BWkDgt1z.js → uniqueVintage-CJXiNNe7.js} +19 -20
  215. package/dist/uniqueVintage-D0jzJWlo.cjs +1213 -0
  216. package/dist/useMessageInterceptor-B87e3yu3.cjs +33 -0
  217. package/dist/useMessageInterceptor-Bb7YRaWk.js +25 -0
  218. package/dist/userIdentityContext-BPqvVIg0.d.cts +20 -0
  219. package/dist/userIdentityContext-wbCRmlzp.d.ts +20 -0
  220. package/dist/utils-C9ZSCx12.js +888 -0
  221. package/dist/utils-D5HO61hG.cjs +1016 -0
  222. package/dist/variantInfo-DbVxA1yE.js +1 -0
  223. package/dist/variantInfo-orXoPBCU.cjs +0 -0
  224. package/dist/venaCbd-Bhhu_qUf.cjs +365 -0
  225. package/dist/venaCbd-CanovPS_.js +365 -0
  226. package/dist/westonJonBoucher-BC0x1ktI.cjs +422 -0
  227. package/dist/{westonJonBoucher-BAGXegsX.js → westonJonBoucher-BUu1_wP1.js} +19 -20
  228. package/dist/wineEnthusiast-BlCryfil.cjs +940 -0
  229. package/dist/{wineEnthusiast-EJbhMeKQ.js → wineEnthusiast-Ck1x5iJq.js} +19 -20
  230. package/dist/wolfMattress-D9Mjq-HP.js +372 -0
  231. package/dist/wolfMattress-JssghhC-.cjs +372 -0
  232. package/dist/wolfTactical-C6exYhL7.cjs +349 -0
  233. package/dist/wolfTactical-CnV3KQdI.js +349 -0
  234. package/package.json +100 -19
  235. package/src/{adapters/spiffy/commerce/api.ts → application/commerce-api.ts} +31 -23
  236. package/src/application/models/colorsConfig.ts +18 -18
  237. package/src/application/models/frontendConfig.ts +3 -3
  238. package/src/application/models/graphql/queries/getMerchantColorsQuery.ts +1 -1
  239. package/src/application/models/graphql/queries/getMerchantFrontendConfigQuery.ts +1 -1
  240. package/src/application/models/index.ts +0 -2
  241. package/src/application/models/localStorageEventListener.ts +2 -2
  242. package/src/application/models/utils/snakeToCamelTransformer.ts +8 -8
  243. package/src/application/models/validators/validateGraphQLColorsConfig.ts +0 -21
  244. package/src/application/models/validators/validateGraphQLFrontendConfig.ts +2 -2
  245. package/src/application/utils/analyticsUtils.ts +53 -22
  246. package/src/application/utils/cdnUtils.ts +11 -0
  247. package/src/atoms/amplitude/amplitudeTrackEventAtom.ts +15 -0
  248. package/src/atoms/app/index.ts +7 -17
  249. package/src/atoms/app/variant.ts +13 -4
  250. package/src/atoms/chat/chatState.ts +28 -18
  251. package/src/atoms/chat/messageQueue.ts +36 -5
  252. package/src/atoms/chat/replies.ts +45 -37
  253. package/src/atoms/envive/enviveConfig.ts +10 -6
  254. package/src/atoms/org/graphqlConfig.ts +7 -1
  255. package/src/atoms/org/index.ts +3 -3
  256. package/src/atoms/org/newOrgConfigAtom.ts +8 -0
  257. package/src/atoms/org/orgPageConfig.ts +6 -5
  258. package/src/atoms/search/chatSearch.ts +11 -5
  259. package/src/atoms/search/searchAPI.ts +22 -12
  260. package/src/atoms/search/searchServiceAdapter.ts +25 -0
  261. package/src/contexts/amplitudeContext.tsx +466 -0
  262. package/src/contexts/cdnContext.tsx +48 -0
  263. package/src/contexts/chatContext.tsx +15 -12
  264. package/src/contexts/enviveConfigContext.tsx +22 -17
  265. package/src/contexts/enviveCssContext.tsx +34 -5
  266. package/src/contexts/featureFlagContext.tsx +193 -0
  267. package/src/contexts/featureFlagServiceContext.tsx +87 -0
  268. package/src/contexts/graphqlContext.tsx +165 -0
  269. package/src/contexts/index.ts +13 -3
  270. package/src/contexts/localStorageContext.tsx +159 -0
  271. package/src/contexts/newOrgConfigContext.tsx +104 -0
  272. package/src/contexts/orgConfigContext.tsx +92 -92
  273. package/src/contexts/searchContext.tsx +187 -0
  274. package/src/contexts/sessionStorageContext.tsx +80 -0
  275. package/src/contexts/shopifyUrlContext.tsx +97 -0
  276. package/src/contexts/types.ts +12 -8
  277. package/src/contexts/userIdentityContext.tsx +197 -0
  278. package/src/events/registerAnalyticsListeners.ts +32 -38
  279. package/src/hooks/index.ts +10 -4
  280. package/src/hooks/useAmplitudeOperations.ts +35 -0
  281. package/src/hooks/useAppDetails.ts +49 -0
  282. package/src/hooks/useCdnOperations.ts +16 -0
  283. package/src/hooks/useChatToggle.ts +5 -4
  284. package/src/hooks/useChatToggleAnalytics.ts +15 -0
  285. package/src/hooks/useGraphQLConfig.ts +63 -0
  286. package/src/hooks/useIdentifyUser.ts +33 -0
  287. package/src/hooks/useImageResolver.ts +7 -7
  288. package/src/hooks/useLocalStorageOperations.ts +92 -0
  289. package/src/hooks/useNewOrgConfig.ts +3 -76
  290. package/src/hooks/useSearch.tsx +21 -18
  291. package/src/hooks/useSearchOperations.ts +97 -0
  292. package/src/hooks/useSessionStorageOperations.ts +28 -0
  293. package/src/hooks/useShopifyUrlOperations.ts +45 -0
  294. package/src/hooks/useTrackComponentVisibleEvent.ts +12 -8
  295. package/src/hooks/useUpdateAnalyticsProps.ts +28 -17
  296. package/src/interceptors/index.ts +0 -1
  297. package/src/interceptors/useMessageInterceptor.ts +5 -8
  298. package/src/merchants/bandolier/bandolier.ts +10 -28
  299. package/src/merchants/carpe/carpe.ts +5 -8
  300. package/src/merchants/coterie/coterie.ts +3 -6
  301. package/src/merchants/default.ts +73 -70
  302. package/src/merchants/domInsertion.ts +30 -0
  303. package/src/merchants/dreamlandBaby/dreamlandBaby.ts +12 -6
  304. package/src/merchants/fiveCbd/fiveCbd.ts +7 -13
  305. package/src/merchants/forLoveAndLemons/forLoveAndLemons.ts +44 -14
  306. package/src/merchants/greenpan/greenpan.ts +6 -12
  307. package/src/merchants/gridInsertion.ts +19 -0
  308. package/src/merchants/grooveLife/grooveLife.ts +13 -7
  309. package/src/merchants/homegrownCannabis/homegrownCannabis.ts +12 -6
  310. package/src/merchants/jackArcher/jackArcher.ts +26 -11
  311. package/src/merchants/jordanCraig/jordanCraig.ts +5 -5
  312. package/src/merchants/kindredBravely/kindredBravely.ts +16 -7
  313. package/src/merchants/kutFromTheKloth/kutFromTheKloth.ts +9 -6
  314. package/src/merchants/larryAndSerges/larryAndSerges.ts +6 -2
  315. package/src/merchants/leapsAndRebounds/leapsAndRebounds.ts +7 -7
  316. package/src/merchants/longevityrx/longevityrx.ts +14 -8
  317. package/src/merchants/lookOptic/lookOptic.ts +6 -6
  318. package/src/merchants/mantraBrand/mantraBrand.ts +14 -8
  319. package/src/merchants/medterra/medterra.ts +14 -11
  320. package/src/merchants/modells/modells.ts +6 -9
  321. package/src/merchants/pressedFloral/pressedFloral.ts +7 -13
  322. package/src/merchants/skinPerfection/skinPerfection.ts +13 -7
  323. package/src/merchants/snapSupplements/snapSupplements.ts +13 -10
  324. package/src/merchants/spanx/spanx.ts +6 -6
  325. package/src/merchants/spanx/spanxStaging.ts +6 -12
  326. package/src/merchants/supergoop/supergoop.ts +11 -12
  327. package/src/merchants/uniqueVintage/uniqueVintage.ts +6 -6
  328. package/src/merchants/venaCbd/venaCbd.ts +6 -9
  329. package/src/merchants/westonJonBoucher/westonJonBoucher.ts +6 -6
  330. package/src/merchants/wineEnthusiast/wineEnthusiast.ts +6 -6
  331. package/src/merchants/wolfMattress/wolfMattress.ts +3 -3
  332. package/src/merchants/wolfTactical/wolfTactical.ts +5 -11
  333. package/src/types/config-versions.ts +6 -0
  334. package/src/types.ts +0 -53
  335. package/dist/adapters/amplitude/index.cjs +0 -14
  336. package/dist/adapters/amplitude/index.d.cts +0 -6
  337. package/dist/adapters/amplitude/index.d.ts +0 -6
  338. package/dist/adapters/amplitude/index.js +0 -12
  339. package/dist/api-B2euFL-5.cjs +0 -269
  340. package/dist/api-XRr_lAG6.js +0 -190
  341. package/dist/application/config/index.cjs +0 -43
  342. package/dist/application/config/index.d.cts +0 -15
  343. package/dist/application/config/index.d.ts +0 -15
  344. package/dist/application/config/index.js +0 -41
  345. package/dist/application/service/customerService/index.cjs +0 -4
  346. package/dist/application/service/customerService/index.d.cts +0 -3
  347. package/dist/application/service/customerService/index.d.ts +0 -3
  348. package/dist/application/service/customerService/index.js +0 -4
  349. package/dist/application/service/index.cjs +0 -42
  350. package/dist/application/service/index.d.cts +0 -413
  351. package/dist/application/service/index.d.ts +0 -413
  352. package/dist/application/service/index.js +0 -12
  353. package/dist/atomStore-ONYy0XuA.d.cts +0 -24
  354. package/dist/atomStore-kOKiEcNl.d.ts +0 -24
  355. package/dist/bandolier-Da4wt6sm.cjs +0 -1230
  356. package/dist/bandolier-DzEmYWcz.js +0 -1230
  357. package/dist/carpe-AXipz0Xl.cjs +0 -608
  358. package/dist/carpe-CaUKwcEa.js +0 -606
  359. package/dist/contexts-CtRlNXaS.js +0 -7674
  360. package/dist/contexts-CtgmnYNn.cjs +0 -9088
  361. package/dist/coterie-oKHAT0lx.js +0 -240
  362. package/dist/customerService-BG1uNZZ1.cjs +0 -36
  363. package/dist/customerService-BHQRnLhC.js +0 -23
  364. package/dist/default-B4fINY5_.cjs +0 -4
  365. package/dist/default-BrTQxA0c.js +0 -4
  366. package/dist/default-C3LrcbZB.cjs +0 -199
  367. package/dist/default-CXkYrLEr.js +0 -176
  368. package/dist/dreamlandBaby-BEqt0eKF.js +0 -347
  369. package/dist/dreamlandBaby-DBDjEJCc.cjs +0 -347
  370. package/dist/featureFlagService-5wdmW02z.d.ts +0 -18
  371. package/dist/featureFlagService-DaelrXEk.d.cts +0 -18
  372. package/dist/fiveCbd-CdqNt16h.cjs +0 -614
  373. package/dist/fiveCbd-D6B-sgnX.js +0 -614
  374. package/dist/forLoveAndLemons-C7GsJG7f.cjs +0 -668
  375. package/dist/forLoveAndLemons-gNDlMtPR.js +0 -666
  376. package/dist/greenpan-B5AaW4M_.js +0 -398
  377. package/dist/greenpan-DrORpYms.cjs +0 -398
  378. package/dist/grooveLife-BJqsfH2H.cjs +0 -343
  379. package/dist/grooveLife-xIUmDM8s.js +0 -343
  380. package/dist/homegrownCannabis-8TZ21u6L.cjs +0 -411
  381. package/dist/homegrownCannabis-BtMuEvbZ.js +0 -411
  382. package/dist/index-CKUpnyJQ.d.ts +0 -72
  383. package/dist/index-DFL1dIT_.d.ts +0 -7
  384. package/dist/index-DXpgMVpp.d.ts +0 -749
  385. package/dist/index-MFbPQ8Ji.d.ts +0 -95
  386. package/dist/index-VHFMGkO-.d.cts +0 -72
  387. package/dist/index-VSFakgAI.d.cts +0 -95
  388. package/dist/index-aNW5V9fh.d.cts +0 -749
  389. package/dist/index-zZjcds15.d.cts +0 -7
  390. package/dist/jackArcher-WtkbLBZj.cjs +0 -728
  391. package/dist/jordanCraig-471FcgqF.cjs +0 -1787
  392. package/dist/jordanCraig-DNOncplU.js +0 -1787
  393. package/dist/kindredBravely-BlLyHGMX.cjs +0 -491
  394. package/dist/kindredBravely-cqZ4OvXp.js +0 -491
  395. package/dist/kutFromTheKloth-3mOIryvt.cjs +0 -370
  396. package/dist/kutFromTheKloth-DtVNCMKa.js +0 -370
  397. package/dist/larryAndSerges-88Bvq-Us.cjs +0 -262
  398. package/dist/larryAndSerges-cvak6May.js +0 -262
  399. package/dist/leapsAndRebounds-Dmf8eUPq.js +0 -361
  400. package/dist/leapsAndRebounds-iWKc923H.cjs +0 -361
  401. package/dist/logger-Dln20ans.cjs +0 -26
  402. package/dist/logger-pdEEY8T2.js +0 -20
  403. package/dist/longevityrx-BTMI9vn-.js +0 -321
  404. package/dist/longevityrx-CobPyigd.cjs +0 -321
  405. package/dist/lookOptic-C4H_c0JZ.cjs +0 -283
  406. package/dist/mantraBrand-CySGqbn6.cjs +0 -751
  407. package/dist/medterra-BnZ5p27n.cjs +0 -584
  408. package/dist/medterra-DaICcPPp.js +0 -584
  409. package/dist/modells-CZ1L6dD_.js +0 -485
  410. package/dist/modells-DF0SndHr.cjs +0 -485
  411. package/dist/orgConfigResults--dAwtw3W.d.ts +0 -881
  412. package/dist/orgConfigResults-BL0XBA6x.d.cts +0 -881
  413. package/dist/pressedFloral-B3t2cYzs.cjs +0 -662
  414. package/dist/pressedFloral-Dsws2Kfb.js +0 -662
  415. package/dist/skinPerfection-CILQM2bR.cjs +0 -335
  416. package/dist/skinPerfection-DmQCntRf.js +0 -335
  417. package/dist/snapSupplements--X_v0KRM.js +0 -286
  418. package/dist/snapSupplements-Djuzl0Ed.cjs +0 -286
  419. package/dist/spanx-Bo81yXSF.cjs +0 -664
  420. package/dist/spanx-DauxB8KE.js +0 -662
  421. package/dist/spanxStaging-BucYQvR1.cjs +0 -849
  422. package/dist/spanxStaging-CfXUukdP.js +0 -846
  423. package/dist/supergoop-8qa_NV3F.cjs +0 -338
  424. package/dist/supergoop-DOaui-A6.js +0 -336
  425. package/dist/types-CD4LFta-.d.cts +0 -33
  426. package/dist/types-CGC6Oozp.cjs +0 -231
  427. package/dist/types-CnTCkyvK.js +0 -177
  428. package/dist/types-DBdI0j89.d.ts +0 -33
  429. package/dist/uniqueVintage-DAne8XcL.cjs +0 -1214
  430. package/dist/variant-CC1nrywd.d.ts +0 -13
  431. package/dist/variant-CQTuQQSq.d.cts +0 -13
  432. package/dist/venaCbd-B1HO_Pkr.cjs +0 -366
  433. package/dist/venaCbd-CnByO-5R.js +0 -366
  434. package/dist/westonJonBoucher-BRfHWMbs.cjs +0 -423
  435. package/dist/wineEnthusiast-DW8JVwV8.cjs +0 -941
  436. package/dist/wolfMattress-B6INZNRJ.cjs +0 -373
  437. package/dist/wolfMattress-D9P7ErH_.js +0 -373
  438. package/dist/wolfTactical-C5Pupi3J.js +0 -350
  439. package/dist/wolfTactical-TMthZM93.cjs +0 -350
  440. package/src/adapters/amplitude/amplitudeAdapter.ts +0 -477
  441. package/src/adapters/amplitude/index.ts +0 -2
  442. package/src/adapters/amplitude/stubAmplitudeAdapter.ts +0 -34
  443. package/src/adapters/spiffy/commerce/graphql.ts +0 -219
  444. package/src/application/config/generalStaticConfig.ts +0 -40
  445. package/src/application/config/index.ts +0 -1
  446. package/src/application/models/domMutationContinuation.ts +0 -7
  447. package/src/application/models/domObservationStrategy.ts +0 -9
  448. package/src/application/service/cachingService.ts +0 -84
  449. package/src/application/service/cdnService.ts +0 -20
  450. package/src/application/service/customerService/index.ts +0 -8
  451. package/src/application/service/customerService/providers/UnsupportedCustomerService.ts +0 -15
  452. package/src/application/service/domMutationObserver.ts +0 -320
  453. package/src/application/service/domMutations/GridInsertionService.ts +0 -123
  454. package/src/application/service/domMutations/dataLayer/dataLayerEventsListener.ts +0 -99
  455. package/src/application/service/domMutations/domInsertionService.ts +0 -90
  456. package/src/application/service/domMutations/domMutationListener.ts +0 -15
  457. package/src/application/service/domMutations/domMutationListenerState.ts +0 -52
  458. package/src/application/service/domMutations/floatingChat/embeddedChatsPlacementsListener.ts +0 -41
  459. package/src/application/service/domMutations/gladly/gladlyListener.ts +0 -61
  460. package/src/application/service/domMutations/spiffy/orgs/common/kustomerVisibilityListener.ts +0 -41
  461. package/src/application/service/domMutations/spiffy/orgs/common/orgsCommonDataLayerListener.ts +0 -119
  462. package/src/application/service/featureFlagService.ts +0 -130
  463. package/src/application/service/graphqlConfigService.ts +0 -59
  464. package/src/application/service/index.ts +0 -32
  465. package/src/application/service/kustomerIntegrationService.ts +0 -111
  466. package/src/application/service/localStorageService.ts +0 -77
  467. package/src/application/service/pageVariantService.ts +0 -866
  468. package/src/application/service/searchService.ts +0 -147
  469. package/src/application/service/sessionStorageService.ts +0 -27
  470. package/src/application/service/shopifyUrlService.ts +0 -63
  471. package/src/application/service/userIdentityService.ts +0 -114
  472. package/src/application/service/windowChatToggleService.ts +0 -80
  473. package/src/application/service/windowDataLayerService.ts +0 -181
  474. package/src/application/service/windowFrontendConfigService.ts +0 -129
  475. package/src/atoms/org/merchantCss.ts +0 -161
  476. package/src/atoms/org/org.ts +0 -251
  477. package/src/atoms/org/orgUIConfig.ts +0 -142
  478. package/src/enabled-features.ts +0 -83
  479. package/src/hooks/useDynamicVariants.ts +0 -226
  480. package/src/hooks/useFileUpload.ts +0 -63
  481. package/src/hooks/useHideElements.ts +0 -85
  482. package/src/interceptors/useFormEscalation.ts +0 -57
  483. /package/dist/{locators-DnKpajbY.js → atomStore-CZnUUsrr.js} +0 -0
  484. /package/dist/{locators-CugndTUM.cjs → atomStore-KSoFS3Jj.cjs} +0 -0
  485. /package/dist/{utilityTypes-C4h2wgAK.cjs → locators-0YYZu9n4.cjs} +0 -0
  486. /package/dist/{utilityTypes-BVikejDo.js → locators-fBXS_pxP.js} +0 -0
  487. /package/dist/{variantInfo-CzhR5W6h.js → utilityTypes-8sETsYPk.js} +0 -0
  488. /package/dist/{variantInfo-CNRTY0gH.cjs → utilityTypes-COShxVir.cjs} +0 -0
  489. /package/src/{application/service/customerService/types.ts → types/customerService.ts} +0 -0
  490. /package/src/{adapters/spiffy/commerce → types}/exceptions/sessionExceptions.ts +0 -0
  491. /package/src/{adapters/spiffy/commerce → types}/exceptions/unsupportedProductExceptions.ts +0 -0
@@ -0,0 +1,2596 @@
1
+ import { FeatureGates, MessageRole, MessageType, ProductExperiment, SpiffyWidgets, getMerchantOrgIdQuery, transformSnakeToCamel, validateGraphQLOrgId, validateOrgConfigResults, validateResponse, validateSuggestion, validateUserEvent } from "./models-UZ6HszfZ.js";
2
+ import { OrgShortName, getOrgInfo } from "./types--pr1GQQx.js";
3
+ import { configVersion, parseHref } from "./graphql-CjB8wHzm.js";
4
+ import { logger_default } from "./api-DjeZXxl_.js";
5
+ import { LocalStorageKeys, baseUrlAtom, cdnUrlAtom, contextSourceAtom, envAtom, orgLevelApiKeyAtom, orgShortNameAtom, reactAppNameAtom, useLocalStorage } from "./enviveConfig-CzhTz8Aa.js";
6
+ import { getAtomStore } from "./atomStore-D8pjE1vL.js";
7
+ import { chatIdAtom, hasParsedVariantInfoAtom, supportedEventAtom, userIdAtom, variantInfoAtom } from "./app-CjsQ2_n-.js";
8
+ import { SpiffyMetricsEventName, useAmplitude, useEnviveConfig } from "./amplitudeContext-hY3caPC6.js";
9
+ import { DOMObserver, coreSupportedEventRequestToApiRequest, messageFromFormSubmittedEvent, messageFromQueryEvent, messageFromResponse, messageFromSuggestionEvent, messageRequestToCommerceMessageRequest } from "./utils-C9ZSCx12.js";
10
+ import { featureFlagServiceAtom, orgIdAtom } from "./graphqlConfig-CYt6tWII.js";
11
+ import { PerfMetricsEvents, chatAtom, chatOnToggleAtom, clearUserEventAtom, createResponsePayload, logPerfMetricAtom, messagesAtom, processUserEventAtom, requestFailureAtom, responseStreamingAtom, suggestionsAtom, suggestionsLoadingAtom, userEventQueueAtom, userEventsAtom, userHasRepliedAtom, userQueueEventCountAtom } from "./chat-hcRc4RRd.js";
12
+ import { newOrgConfigAtom } from "./org-xMytX--e.js";
13
+ import { ProductSorting, addSearchFilterAtom, chatSearchIsLoadingAtom, chatSearchProductSortingAtom, chatSearchProducts, chatSearchStateAtom, clearSearchFiltersAtom, clearSearchServiceFunction, createFilterOption, filteredSearchProductsAtom, formatFilterDisplayName, handleSearchResultsAtom, performSearchAtom, removeSearchFilterAtom, searchAtom, searchFiltersAtom, searchParamsAtom, searchProductSortingAtom, searchSelectedFiltersAtom, setSearchServiceFunction } from "./search-BMOAmrmY.js";
14
+ import { autocompleteStateAtom, isFilterOpenAtom } from "./globalSearch-B6THR7Jx.js";
15
+ import { useMessageInterceptor } from "./useMessageInterceptor-Bb7YRaWk.js";
16
+ import React, { createContext, useCallback, useContext, useEffect, useMemo, useRef, useState } from "react";
17
+ import { useAtom, useAtomValue, useSetAtom } from "jotai";
18
+ import { Configuration, ContextEnvEnum, ContextSourceEnum, CustomerServiceApi, DefaultApi, FormType, InferenceApi, PLPAttributeCategory, ResponseCategory, ResponseError, SearchApi, UserEventCategory, V1OrgConfigGetSourceEnum } from "@spiffy-ai/commerce-api-client";
19
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
20
+ import { v4 } from "uuid";
21
+ import UAParser from "ua-parser-js";
22
+
23
+ //#region src/contexts/cdnContext.tsx
24
+ const CdnContext = createContext(null);
25
+ const CdnProvider = ({ children }) => {
26
+ const cdnUrl = useAtomValue(cdnUrlAtom) || "https://cdn.spiffy.ai/other";
27
+ const getCdnBasePath = useCallback(() => {
28
+ return cdnUrl;
29
+ }, [cdnUrl]);
30
+ const getAssetURL = useCallback((assetName, orgShortName) => {
31
+ return `${getCdnBasePath()}/assets/${orgShortName}/${assetName}`;
32
+ }, [getCdnBasePath]);
33
+ const value = useMemo(() => ({
34
+ cdnUrl,
35
+ getCdnBasePath,
36
+ getAssetURL
37
+ }), [
38
+ cdnUrl,
39
+ getCdnBasePath,
40
+ getAssetURL
41
+ ]);
42
+ return /* @__PURE__ */ jsx(CdnContext.Provider, {
43
+ value,
44
+ children
45
+ });
46
+ };
47
+ const useCdn = () => {
48
+ const context = useContext(CdnContext);
49
+ if (!context) throw new Error("useCdn must be used within a CdnProvider");
50
+ return context;
51
+ };
52
+
53
+ //#endregion
54
+ //#region src/types/exceptions/sessionExceptions.ts
55
+ var SessionRestartRequired = class extends Error {
56
+ constructor() {
57
+ super("Session restart required");
58
+ this.name = "SessionRestartRequired";
59
+ }
60
+ };
61
+
62
+ //#endregion
63
+ //#region src/types/exceptions/unsupportedProductExceptions.ts
64
+ var UnsupportedProductException = class extends Error {
65
+ constructor() {
66
+ super("Unsupported product");
67
+ this.name = "UnsupportedProduct";
68
+ }
69
+ };
70
+
71
+ //#endregion
72
+ //#region src/application/commerce-api.ts
73
+ async function errorResponseBody$1(error) {
74
+ try {
75
+ return await error.response.json();
76
+ } catch {
77
+ return {};
78
+ }
79
+ }
80
+ async function throwSessionRestartRequiredIf$1(errorMsg, error) {
81
+ if (!(error instanceof ResponseError)) {
82
+ logger_default.logInfo(errorMsg, error);
83
+ throw error;
84
+ }
85
+ const errorResponse = await errorResponseBody$1(error);
86
+ if (errorResponse?.message?.toLowerCase() === "unsupported product" || errorResponse?.app_code?.toUpperCase() === "PRODUCT_NOT_FOUND") throw new UnsupportedProductException();
87
+ else if (errorResponse?.app_code?.toUpperCase() === "RESTART_SESSION" || errorResponse?.sub_code?.toUpperCase() === "NOT_FOUND") {
88
+ logger_default.logInfo("Session does not exist. Re-start session", error, error.response, errorResponse);
89
+ throw new SessionRestartRequired();
90
+ }
91
+ logger_default.logInfo(errorMsg, error);
92
+ throw error;
93
+ }
94
+ var CommerceApiClient = class CommerceApiClient {
95
+ static {
96
+ this.getInstance = () => {
97
+ if (!CommerceApiClient.instance) CommerceApiClient.instance = new CommerceApiClient();
98
+ return CommerceApiClient.instance;
99
+ };
100
+ }
101
+ constructor(basePath) {
102
+ this.suggestionsAbortController = new AbortController();
103
+ this.responsesAbortController = new AbortController();
104
+ const baseUrl = getAtomStore().get(baseUrlAtom);
105
+ const config = new Configuration({
106
+ basePath: basePath || baseUrl,
107
+ headers: {
108
+ "Content-Type": "application/json",
109
+ Accept: "application/json"
110
+ }
111
+ });
112
+ this.defaultApi = new DefaultApi(config);
113
+ this.inferenceApi = new InferenceApi(config);
114
+ this.customerServiceApi = new CustomerServiceApi(config);
115
+ }
116
+ static {
117
+ this.resolveUrl = async (url) => {
118
+ const atomStore = getAtomStore();
119
+ const orgShortName = atomStore.get(orgShortNameAtom);
120
+ const orgId = atomStore.get(orgIdAtom);
121
+ const userId = atomStore.get(userIdAtom);
122
+ const chatId = atomStore.get(chatIdAtom);
123
+ const source = atomStore.get(contextSourceAtom);
124
+ const env = atomStore.get(envAtom);
125
+ const featureFlagService = atomStore.get(featureFlagServiceAtom);
126
+ const context = {
127
+ user_id: userId ?? "",
128
+ org_id: orgId ?? "",
129
+ org_short_name: orgShortName ?? "",
130
+ chat_id: chatId ?? "",
131
+ source: source ?? ContextSourceEnum.App,
132
+ env: env ?? ContextEnvEnum.Dev
133
+ };
134
+ const featureGates = featureFlagService?.featureFlagService?.getFeatureFlags() || {};
135
+ const urlResolvingRequest = {
136
+ url,
137
+ context,
138
+ feature_gates: featureGates
139
+ };
140
+ return await (await CommerceApiClient.getInstance().inferenceApi.v1UrlResolvingPostRaw({ UrlResolvingRequest: urlResolvingRequest })).raw.json();
141
+ };
142
+ }
143
+ static {
144
+ this.reportSession = async (reportRequest) => {
145
+ await CommerceApiClient.getInstance().defaultApi.v1ChatsReportSessionIdPost({ ReportSessionRequest: reportRequest });
146
+ };
147
+ }
148
+ static {
149
+ this.getNextResponses = async (payload) => {
150
+ try {
151
+ return (await CommerceApiClient.getInstance().inferenceApi.v1NextResponsesPost({ NextMessageRequest: messageRequestToCommerceMessageRequest(payload) })).map((resp) => validateResponse(resp)).map((resp) => messageFromResponse(resp)).filter((m) => m != null);
152
+ } catch (err) {
153
+ logger_default.logInfo("Failed to get next responses", err, {
154
+ payloadContext: payload?.context,
155
+ userEvents: payload?.userEvents
156
+ });
157
+ await throwSessionRestartRequiredIf$1("Failed to get next responses", err);
158
+ return [];
159
+ }
160
+ };
161
+ }
162
+ static {
163
+ this.getNextResponseStreaming = (payload) => {
164
+ async function* generate(inferenceApi, abortController) {
165
+ try {
166
+ const response = await inferenceApi.v1NextResponsesPostRaw({ NextMessageRequest: messageRequestToCommerceMessageRequest(payload) }, { signal: abortController.signal });
167
+ if (!response.raw.body) {
168
+ logger_default.logError("[spiffy-ai] No body in the streamed response", void 0, { response: response.raw });
169
+ return;
170
+ }
171
+ const reader = response.raw.body.getReader();
172
+ const decoder = new TextDecoder("utf-8");
173
+ let partial = "";
174
+ const safeParse = (line) => {
175
+ try {
176
+ return JSON.parse(line);
177
+ } catch (err) {
178
+ logger_default.logError("[spiffy-ai] Error parsing streamed line", err, {
179
+ line,
180
+ partial
181
+ });
182
+ partial = line;
183
+ return partial;
184
+ }
185
+ };
186
+ const processChunk = (chunk) => {
187
+ return `${partial}${chunk}`.split("\n").map((line) => line.replace(/^data: /, "").trim()).filter((line) => line !== "" && line !== "[DONE]").map(safeParse).filter((v) => v);
188
+ };
189
+ while (true) {
190
+ const { done, value } = await reader.read();
191
+ if (done) break;
192
+ const chunk = decoder.decode(value);
193
+ const parsedLines = processChunk(chunk);
194
+ for (const parsedLine of parsedLines) {
195
+ const validatedResponse = validateResponse(parsedLine);
196
+ if (validatedResponse) yield validatedResponse;
197
+ }
198
+ }
199
+ } catch (error) {
200
+ logger_default.logError("[spiffy-ai] Failed to get next streaming responses", error, {
201
+ payloadContext: payload?.context,
202
+ userEvents: payload?.userEvents
203
+ });
204
+ await throwSessionRestartRequiredIf$1("Failed to get next streaming responses", error);
205
+ }
206
+ }
207
+ CommerceApiClient.getInstance().responsesAbortController.abort();
208
+ CommerceApiClient.getInstance().responsesAbortController = new AbortController();
209
+ return generate(CommerceApiClient.getInstance().inferenceApi, CommerceApiClient.getInstance().responsesAbortController);
210
+ };
211
+ }
212
+ static {
213
+ this.getNextSuggestions = async (payload) => {
214
+ try {
215
+ CommerceApiClient.getInstance().suggestionsAbortController.abort();
216
+ CommerceApiClient.getInstance().suggestionsAbortController = new AbortController();
217
+ return (await CommerceApiClient.getInstance().inferenceApi.v1NextSuggestionsPost({ NextMessageRequest: messageRequestToCommerceMessageRequest(payload) }, { signal: CommerceApiClient.getInstance().suggestionsAbortController.signal })).map((resp) => validateSuggestion(resp)).filter((suggestion) => suggestion != null);
218
+ } catch (error) {
219
+ logger_default.logInfo("Failed to get suggestions", error, {
220
+ payloadContext: payload?.context,
221
+ userEvents: payload?.userEvents
222
+ });
223
+ await throwSessionRestartRequiredIf$1("Failed to get suggestions", error);
224
+ return [];
225
+ }
226
+ };
227
+ }
228
+ static {
229
+ this.getResponses = async (orgId, chatId, userId) => {
230
+ let data = {
231
+ responses: [],
232
+ suggestions: [],
233
+ user_events: []
234
+ };
235
+ const request = {
236
+ org_id: orgId,
237
+ chat_id: chatId,
238
+ user_id: userId
239
+ };
240
+ try {
241
+ data = await CommerceApiClient.getInstance().defaultApi.v1GetSessionMessages(request);
242
+ } catch (error) {
243
+ await throwSessionRestartRequiredIf$1("Failed to get chat responses", error);
244
+ }
245
+ const responses = data?.responses?.map((turn) => turn.map((response) => validateResponse(response)).filter((response) => response != null));
246
+ const suggestions = data?.suggestions.map((suggestion) => validateSuggestion(suggestion)).filter((suggestion) => suggestion != null);
247
+ const userEvents = data?.user_events.map((event) => validateUserEvent(event)).filter((event) => event != null);
248
+ const formSubmittedUserEventsFormIds = userEvents.filter((event) => event.category === UserEventCategory.FormSubmitted).map((event) => event.attributes.formResponseId);
249
+ const assistantMessages = responses.map((turn) => turn.filter((response) => !(response.category === ResponseCategory.Form && formSubmittedUserEventsFormIds.includes(response.id))).map((response) => messageFromResponse(response)).filter((message) => message != null)).filter((turn) => turn.length > 0);
250
+ const userMessages = userEvents.map((event) => {
251
+ if ([UserEventCategory.QueryTyped, UserEventCategory.Search].includes(event.category)) return [messageFromQueryEvent(event)];
252
+ if (event.category === UserEventCategory.SuggestionClicked) return [messageFromSuggestionEvent(event, suggestions)];
253
+ if (event.category === UserEventCategory.FormSubmitted) {
254
+ const formResponse = responses.flat().find((response) => response.id === event.attributes.formResponseId && event.attributes.formType !== FormType.Escalation);
255
+ if (formResponse && formResponse.category === ResponseCategory.Form) return [messageFromFormSubmittedEvent(event, formResponse.attributes)];
256
+ }
257
+ return [];
258
+ }).filter((message) => message.length > 0);
259
+ const sortedMessages = [...assistantMessages, ...userMessages].sort((a, b) => new Date(a[0].createdAt).getTime() - new Date(b[0].createdAt).getTime());
260
+ return {
261
+ responses,
262
+ userEvents,
263
+ suggestions,
264
+ messages: sortedMessages
265
+ };
266
+ };
267
+ }
268
+ static {
269
+ this.isSupportedEvent = async (payload) => {
270
+ try {
271
+ const httpResponseText = await (await CommerceApiClient.getInstance().inferenceApi.v1SupportedEventPostRaw({ SupportedEventRequest: coreSupportedEventRequestToApiRequest(payload) })).raw.text();
272
+ const httpResponseJson = JSON.parse(httpResponseText);
273
+ return {
274
+ ...httpResponseJson,
275
+ numberOfReviews: httpResponseJson.num_of_reviews,
276
+ merchant_tags: httpResponseJson.merchant_tags || []
277
+ };
278
+ } catch (err) {
279
+ logger_default.logError("Failed to get response for v1SupportedEventPost", { err });
280
+ return {
281
+ supported: false,
282
+ ready: false,
283
+ category: void 0,
284
+ collections: [],
285
+ numberOfReviews: void 0,
286
+ top_category: void 0,
287
+ merchant_tags: []
288
+ };
289
+ }
290
+ };
291
+ }
292
+ static {
293
+ this.identifyUser = async (spiffyUserId, merchantUserId, uaDetails) => {
294
+ try {
295
+ await CommerceApiClient.getInstance().defaultApi.v1AnalyticsIdentifyPost({ AnalyticsIdentifyRequest: {
296
+ user_id: spiffyUserId,
297
+ os_name: uaDetails.os,
298
+ os_version: uaDetails.osVersion,
299
+ platform: uaDetails.os,
300
+ device_id: uaDetails.deviceModel,
301
+ device_brand: uaDetails.deviceBrand,
302
+ device_manufacturer: uaDetails.deviceManufacturer,
303
+ device_model: uaDetails.deviceModel,
304
+ user_properties: {
305
+ cdp_user_id: merchantUserId,
306
+ browser: uaDetails.browser,
307
+ browser_version: uaDetails.browserVersion,
308
+ user_agent: uaDetails.userAgent
309
+ }
310
+ } });
311
+ } catch (err) {
312
+ logger_default.logError("Failed to identify user", err);
313
+ }
314
+ };
315
+ }
316
+ static {
317
+ this.mapContextSourceToV1OrgConfigGetSource = (source) => {
318
+ if (source === void 0) return void 0;
319
+ switch (source) {
320
+ case ContextSourceEnum.Fork: return V1OrgConfigGetSourceEnum.Fork;
321
+ case ContextSourceEnum.Playground: return V1OrgConfigGetSourceEnum.Playground;
322
+ case ContextSourceEnum.App: return V1OrgConfigGetSourceEnum.App;
323
+ case ContextSourceEnum.Test: return V1OrgConfigGetSourceEnum.Test;
324
+ default: return source;
325
+ }
326
+ };
327
+ }
328
+ static {
329
+ this.getOrgConfig = async (user_id) => {
330
+ try {
331
+ const atomStore = getAtomStore();
332
+ const reactAppName = atomStore.get(reactAppNameAtom);
333
+ const contextSource = atomStore.get(contextSourceAtom);
334
+ const featureFlagService = atomStore.get(featureFlagServiceAtom);
335
+ const request = {
336
+ namespace: reactAppName,
337
+ user_id,
338
+ source: this.mapContextSourceToV1OrgConfigGetSource(contextSource),
339
+ include_experiments: Object.values(ProductExperiment),
340
+ include_feature_gates: Object.entries(featureFlagService?.featureFlagService?.getFeatureFlags() || {}).filter(([, isEnabled]) => isEnabled).map(([featureGateName]) => featureGateName)
341
+ };
342
+ const response = await CommerceApiClient.getInstance().defaultApi.v1OrgConfigGet(request);
343
+ return validateOrgConfigResults(response);
344
+ } catch (err) {
345
+ logger_default.logError(`Failed to get org config`, err, { err });
346
+ return;
347
+ }
348
+ };
349
+ }
350
+ static {
351
+ this.addNoteToLatestConversation = async (spiffyUserId, email, customerServiceProvider) => {
352
+ logger_default.logInfo(`addNoteToLatestConversation - user_id=${spiffyUserId} email=${email} customer_service_provider=${customerServiceProvider}`);
353
+ try {
354
+ await CommerceApiClient.getInstance().customerServiceApi.v1CustserviceAddNoteToLatestConversationPost({ AddNoteToLatestConversationRequest: {
355
+ spiffy_user_id: spiffyUserId,
356
+ email,
357
+ customer_service_provider: customerServiceProvider
358
+ } });
359
+ } catch (err) {
360
+ logger_default.logError("Failed to add note to latest conversation", { err });
361
+ }
362
+ };
363
+ }
364
+ static {
365
+ this.getCustomerServiceApi = () => CommerceApiClient.getInstance().customerServiceApi;
366
+ }
367
+ };
368
+ var commerce_api_default = CommerceApiClient;
369
+
370
+ //#endregion
371
+ //#region src/hooks/useAmplitudeOperations.ts
372
+ const useAmplitudeTracking = () => {
373
+ const { trackEvent, isReady } = useAmplitude();
374
+ const [loading, setLoading] = useState(false);
375
+ const [error, setError] = useState(null);
376
+ return {
377
+ track: useCallback(async (eventName, eventProps) => {
378
+ if (!isReady) return;
379
+ setLoading(true);
380
+ setError(null);
381
+ try {
382
+ await trackEvent({
383
+ eventName,
384
+ eventProps
385
+ });
386
+ } catch (err) {
387
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Tracking failed"));
388
+ throw err;
389
+ } finally {
390
+ setLoading(false);
391
+ }
392
+ }, [trackEvent, isReady]),
393
+ loading,
394
+ error,
395
+ isReady
396
+ };
397
+ };
398
+
399
+ //#endregion
400
+ //#region src/hooks/useBlockBackButton.ts
401
+ const useBlockBackButton = (enabled, callback) => {
402
+ useEffect(() => {
403
+ if (enabled && window) {
404
+ if (window.history.scrollRestoration) window.history.scrollRestoration = "manual";
405
+ window.history.pushState(null, document.title, window.location.href);
406
+ window.onpopstate = (e) => {
407
+ e.preventDefault();
408
+ window.history.pushState(null, document.title, window.location.href);
409
+ callback?.();
410
+ };
411
+ }
412
+ return () => {
413
+ if (enabled && window) {
414
+ window.history.back();
415
+ window.onpopstate = null;
416
+ window.history.scrollRestoration = "auto";
417
+ }
418
+ };
419
+ }, [enabled]);
420
+ };
421
+
422
+ //#endregion
423
+ //#region src/hooks/useCdnOperations.ts
424
+ const useCdnUrl = () => {
425
+ const { cdnUrl } = useCdn();
426
+ return cdnUrl;
427
+ };
428
+ const useCdnBasePath = () => {
429
+ const { getCdnBasePath } = useCdn();
430
+ return getCdnBasePath();
431
+ };
432
+ const useAssetUrl = (assetName, orgShortName) => {
433
+ const { getAssetURL } = useCdn();
434
+ return getAssetURL(assetName, orgShortName);
435
+ };
436
+
437
+ //#endregion
438
+ //#region src/hooks/useChatToggle.ts
439
+ const useChatToggle = () => {
440
+ const onToggle = useSetAtom(chatOnToggleAtom);
441
+ const { isOpen } = useAtomValue(chatAtom);
442
+ const { trackEvent } = useAmplitude();
443
+ const toggle = (triggerLocation, triggerId) => {
444
+ if (!isOpen) trackEvent({
445
+ eventName: SpiffyMetricsEventName.ChatComponentExpanded,
446
+ eventProps: { message_metadata: {
447
+ trigger_location: triggerLocation,
448
+ trigger_id: triggerId
449
+ } }
450
+ });
451
+ else trackEvent({
452
+ eventName: SpiffyMetricsEventName.ChatComponentCollapsed,
453
+ eventProps: { message_metadata: {
454
+ trigger_location: triggerLocation,
455
+ trigger_id: triggerId
456
+ } }
457
+ });
458
+ onToggle();
459
+ };
460
+ const openChat = (triggerLocation, triggerId) => {
461
+ if (!isOpen) toggle(triggerLocation, triggerId);
462
+ };
463
+ const closeChat = (triggerLocation, triggerId) => {
464
+ if (isOpen) toggle(triggerLocation, triggerId);
465
+ };
466
+ return {
467
+ toggle,
468
+ isOpen,
469
+ openChat,
470
+ closeChat
471
+ };
472
+ };
473
+
474
+ //#endregion
475
+ //#region src/hooks/useChatToggleAnalytics.ts
476
+ const useChatToggleAnalytics = () => {
477
+ const setChatOnToggle = useSetAtom(chatOnToggleAtom);
478
+ const { track } = useAmplitudeTracking();
479
+ const toggleChat = (triggerLocation) => {
480
+ setChatOnToggle(triggerLocation, track);
481
+ };
482
+ return { toggleChat };
483
+ };
484
+
485
+ //#endregion
486
+ //#region src/hooks/useCustomerSupportHandoff.ts
487
+ /**
488
+ * Hook to call the `click` method of the merchant's customer support chat widget.
489
+ *
490
+ * @param onSwitchToAgent a function to override the function returned by the hook. This is mainly to
491
+ * preserve backward compatibility for merchants not using Kustomer and will be removed when all
492
+ * CS integrations are handled.
493
+ *
494
+ * @returns a function that searches for the customer support chat widget and calls the `click` method.
495
+ */
496
+ const useCustomerSupportHandoff = (onSwitchToAgent) => {
497
+ const onKustomerSwitch = useCallback(() => {
498
+ const kustomerElement = document.getElementById("kustomer-ui-sdk-iframe");
499
+ if (kustomerElement == null || !(kustomerElement instanceof HTMLIFrameElement)) {
500
+ logger_default.logError("[spiffy-ai] Kustomer iFrame element not found", void 0);
501
+ return;
502
+ }
503
+ const kustomerButton = kustomerElement.contentWindow?.document?.getElementById("rootChatIcon");
504
+ if (kustomerButton == null) {
505
+ logger_default.logError("[spiffy-ai] Kustomer button not found", void 0);
506
+ return;
507
+ }
508
+ kustomerButton.click();
509
+ }, []);
510
+ if (onSwitchToAgent != null) return { onSwitch: onSwitchToAgent };
511
+ return { onSwitch: onKustomerSwitch };
512
+ };
513
+
514
+ //#endregion
515
+ //#region src/hooks/useDebounce.ts
516
+ function useDebounce(value, delay) {
517
+ const [debouncedValue, setDebouncedValue] = useState(value);
518
+ useEffect(() => {
519
+ const handler = setTimeout(() => {
520
+ setDebouncedValue(value);
521
+ }, delay);
522
+ return () => {
523
+ clearTimeout(handler);
524
+ };
525
+ }, [value, delay]);
526
+ return debouncedValue;
527
+ }
528
+
529
+ //#endregion
530
+ //#region src/hooks/useElementObserver.ts
531
+ const useElementObserver = (selector) => {
532
+ const INITIAL_RENDER_STATE = true;
533
+ const eoRef = useRef(DOMObserver.add(selector));
534
+ const [renderBlocked, setRenderBlocked] = useState(INITIAL_RENDER_STATE);
535
+ /**
536
+ * Fired every time the HTML element changes.
537
+ *
538
+ * @param fn
539
+ */
540
+ const onChange = (fn) => {
541
+ eoRef.current?.registerOnChange(fn);
542
+ };
543
+ /**
544
+ * Fired when the HTML element is added to the DOM.
545
+ *
546
+ * @param fn
547
+ */
548
+ const onAdd = (fn) => {
549
+ eoRef.current?.registerOnAdd(fn);
550
+ };
551
+ /**
552
+ * Fired when the HTML element is removed from the DOM.
553
+ *
554
+ * @param fn
555
+ */
556
+ const onRemove = (fn) => {
557
+ eoRef.current?.registerOnRemove(fn);
558
+ };
559
+ /**
560
+ * Fired when the class of the HTML element changes.
561
+ *
562
+ * @param fn
563
+ */
564
+ const onClassChange = (fn) => {
565
+ eoRef.current?.registerOnclassChange(fn);
566
+ };
567
+ /**
568
+ * Fired when a class is added to the HTML element.
569
+ *
570
+ * @param className
571
+ * @param fn
572
+ */
573
+ const onClassAdded = (className, fn) => {
574
+ eoRef.current?.registerOnClassAdded(className, fn);
575
+ };
576
+ /**
577
+ * Fired when a class is removed from the HTML element.
578
+ *
579
+ * @param className
580
+ * @param fn
581
+ */
582
+ const onClassRemoved = (className, fn) => {
583
+ eoRef.current?.registerOnClassRemoved(className, fn);
584
+ };
585
+ /**
586
+ * Fired when a child element is added to the HTML element.
587
+ *
588
+ * @param fn
589
+ */
590
+ const onAddChild = (fn) => {
591
+ eoRef.current?.registerOnAddChild(fn);
592
+ };
593
+ /**
594
+ * Fired when a child element is removed from the HTML element.
595
+ *
596
+ * @param fn
597
+ */
598
+ const onRemoveChild = (fn) => {
599
+ eoRef.current?.registerOnRemoveChild(fn);
600
+ };
601
+ /**
602
+ * Allows hooking event listeners to the HTML element, such as `focus`, `blur`, etc.
603
+ *
604
+ * @param event
605
+ * @param fn
606
+ */
607
+ const onEvent = (event, fn) => {
608
+ eoRef.current.registerEvent(event, fn);
609
+ };
610
+ /**
611
+ * Useful when rendering a React.js component inside the HTML element.
612
+ *
613
+ * @param fn
614
+ * @returns
615
+ */
616
+ const render = (fn) => {
617
+ if (!renderBlocked) return eoRef.current.render(fn);
618
+ };
619
+ /**
620
+ * Checks if the element exists in the DOM.
621
+ *
622
+ * @returns
623
+ */
624
+ const exists = () => !!eoRef.current.getNode();
625
+ /**
626
+ * Checks if rendering is unblocked.
627
+ *
628
+ * @returns
629
+ */
630
+ const isRendered = () => !renderBlocked;
631
+ /**
632
+ * Triggers an event for the HTML element.
633
+ *
634
+ * @param event
635
+ */
636
+ const fire = (event) => {
637
+ eoRef.current.fire(event);
638
+ };
639
+ /**
640
+ * Shows the HTML element.
641
+ *
642
+ * @returns
643
+ */
644
+ const show = () => eoRef.current.show();
645
+ /**
646
+ * Hides the HTML element.
647
+ *
648
+ * @returns
649
+ */
650
+ const hide = () => eoRef.current.hide();
651
+ /**
652
+ * Blocks the rendering of elements.
653
+ *
654
+ * @returns
655
+ */
656
+ const blockRendering = () => setRenderBlocked(true);
657
+ /**
658
+ * Unblocks the rendering of elements.
659
+ *
660
+ * @returns
661
+ */
662
+ const unblockRendering = () => setRenderBlocked(false);
663
+ /**
664
+ * Applies CSS styles to the HTML element.
665
+ *
666
+ * @param styles
667
+ */
668
+ const applyStyle = (styles) => {
669
+ const node = eoRef?.current?.getNode();
670
+ node && Object.assign(node.style, styles);
671
+ };
672
+ useEffect(() => {
673
+ eoRef.current.init();
674
+ eoRef.current.registerOnReset(() => setRenderBlocked(INITIAL_RENDER_STATE));
675
+ DOMObserver.observe();
676
+ return () => DOMObserver.remove(selector);
677
+ }, [selector.getPattern()]);
678
+ return {
679
+ targetNode: eoRef.current.getNode(),
680
+ onChange,
681
+ onAdd,
682
+ onRemove,
683
+ onClassChange,
684
+ onClassAdded,
685
+ onClassRemoved,
686
+ onAddChild,
687
+ onRemoveChild,
688
+ onEvent,
689
+ blockRendering,
690
+ unblockRendering,
691
+ exists,
692
+ isRendered,
693
+ render,
694
+ fire,
695
+ show,
696
+ hide,
697
+ applyStyle
698
+ };
699
+ };
700
+
701
+ //#endregion
702
+ //#region src/hooks/useGrabAndScroll.ts
703
+ const animateHorizontalScroll = ({ element, duration, targetScroll, multiply = 1, direction, callback, offset = 0 }) => {
704
+ const start = element.scrollLeft;
705
+ const distance = (targetScroll - start) * multiply;
706
+ const startTime = performance.now();
707
+ function easeOutSine(x) {
708
+ return Math.sin(x * Math.PI / 2);
709
+ }
710
+ function scrollStep(currentTime) {
711
+ const timeElapsed = currentTime - startTime;
712
+ const progress = Math.min(timeElapsed / duration, 1);
713
+ const easing = easeOutSine(progress);
714
+ const step = start + distance * easing;
715
+ const canScroll = (direction === "rt" ? element.scrollLeft < step : element.scrollLeft > step) || !direction;
716
+ if (step > 0 && canScroll) element.scrollTo(step, 0);
717
+ if (timeElapsed < duration) requestAnimationFrame(scrollStep);
718
+ else if (element.scrollLeft + offset === element.scrollWidth) callback?.("rt");
719
+ else if (element.scrollLeft <= 1) callback?.("lt");
720
+ else callback?.("ct");
721
+ }
722
+ requestAnimationFrame(scrollStep);
723
+ };
724
+ const useGrabAndScroll = (enabled, chunkWidth, speed = 400, offset = 0) => {
725
+ const containerRef = useRef(null);
726
+ const [leftArrow, setLeftArrow] = useState(false);
727
+ const [rightArrow, setRightArrow] = useState(true);
728
+ const handleArrows = (position) => {
729
+ switch (position) {
730
+ case "lt":
731
+ setLeftArrow(false);
732
+ setRightArrow(true);
733
+ break;
734
+ case "rt":
735
+ setLeftArrow(true);
736
+ setRightArrow(false);
737
+ break;
738
+ default:
739
+ setLeftArrow(true);
740
+ setRightArrow(true);
741
+ }
742
+ };
743
+ const animationTrigger = () => {
744
+ if (enabled && containerRef.current) {
745
+ const dist = (containerRef?.current?.scrollLeft || 0) / chunkWidth;
746
+ const targetScroll = chunkWidth * (Math.floor(dist) + (dist % 1 > .5 ? 1 : 0));
747
+ animateHorizontalScroll({
748
+ element: containerRef.current,
749
+ targetScroll,
750
+ duration: speed,
751
+ offset,
752
+ callback: handleArrows
753
+ });
754
+ }
755
+ };
756
+ const onNext = (cardsToSlide) => {
757
+ if (containerRef.current) {
758
+ const targetScroll = containerRef.current.scrollLeft + chunkWidth;
759
+ animateHorizontalScroll({
760
+ element: containerRef.current,
761
+ targetScroll,
762
+ duration: speed,
763
+ direction: "rt",
764
+ multiply: cardsToSlide,
765
+ offset,
766
+ callback: handleArrows
767
+ });
768
+ }
769
+ };
770
+ const onPrevious = (cardsToSlide) => {
771
+ if (containerRef.current) {
772
+ const targetScroll = containerRef.current.scrollLeft - chunkWidth;
773
+ animateHorizontalScroll({
774
+ element: containerRef.current,
775
+ targetScroll,
776
+ duration: speed,
777
+ direction: "lt",
778
+ multiply: cardsToSlide,
779
+ offset,
780
+ callback: handleArrows
781
+ });
782
+ }
783
+ };
784
+ return {
785
+ containerRef,
786
+ leftArrow,
787
+ rightArrow,
788
+ animationTrigger,
789
+ onNext,
790
+ onPrevious
791
+ };
792
+ };
793
+
794
+ //#endregion
795
+ //#region src/contexts/graphqlContext.tsx
796
+ const GraphQLContext = createContext(null);
797
+ const colorsAndFrontendConfigQuery = () => `
798
+ query ($version: String = "${configVersion()}") {
799
+ me {
800
+ getProductsConfigByVersion(version: $version) {
801
+ frontend { values }
802
+ colors { values }
803
+ }
804
+ }
805
+ }
806
+ `;
807
+ const GraphQLProvider = ({ children }) => {
808
+ const apiKey = useAtomValue(orgLevelApiKeyAtom);
809
+ const baseUrl = useAtomValue(baseUrlAtom);
810
+ const isReady = Boolean(apiKey && baseUrl);
811
+ const executeQuery = useCallback(async (query, variables) => {
812
+ if (!isReady) throw new Error("GraphQL client not ready - missing apiKey or baseUrl");
813
+ const response = await fetch(`${baseUrl}/v1/graphql`, {
814
+ method: "POST",
815
+ headers: {
816
+ "Content-Type": "application/json",
817
+ Authorization: `Bearer ${apiKey}`
818
+ },
819
+ body: JSON.stringify({
820
+ query,
821
+ variables
822
+ })
823
+ });
824
+ if (!response.ok) throw new Error(`GraphQL request failed: ${response.statusText}`);
825
+ const result = await response.json();
826
+ if (result.errors) throw new Error(`GraphQL errors: ${JSON.stringify(result.errors)}`);
827
+ return result.data;
828
+ }, [
829
+ apiKey,
830
+ baseUrl,
831
+ isReady
832
+ ]);
833
+ const getOrgId = useCallback(async () => {
834
+ const response = await executeQuery(getMerchantOrgIdQuery);
835
+ return validateGraphQLOrgId(response.me.org?.id);
836
+ }, [executeQuery]);
837
+ const getColorsAndFrontendConfig = useCallback(async () => {
838
+ try {
839
+ const query = await colorsAndFrontendConfigQuery();
840
+ if (!query) throw new Error("Colors and frontend config query is not defined");
841
+ const response = await executeQuery(query);
842
+ const colorsConfig = response.me.getProductsConfigByVersion?.colors?.values;
843
+ const frontendConfig = response.me.getProductsConfigByVersion?.frontend?.values;
844
+ const transformedColorConfig = transformSnakeToCamel(colorsConfig);
845
+ const transformedFrontendConfig = transformSnakeToCamel(frontendConfig);
846
+ return {
847
+ colorsConfig: transformedColorConfig,
848
+ frontendConfig: transformedFrontendConfig
849
+ };
850
+ } catch (err) {
851
+ logger_default.logError("Error fetching graphql colors and frontend config", err);
852
+ return {
853
+ colorsConfig: void 0,
854
+ frontendConfig: void 0
855
+ };
856
+ }
857
+ }, [executeQuery]);
858
+ const value = useMemo(() => ({
859
+ executeQuery,
860
+ getOrgId,
861
+ getColorsAndFrontendConfig,
862
+ isReady
863
+ }), [
864
+ executeQuery,
865
+ getOrgId,
866
+ getColorsAndFrontendConfig,
867
+ isReady
868
+ ]);
869
+ return /* @__PURE__ */ jsx(GraphQLContext.Provider, {
870
+ value,
871
+ children
872
+ });
873
+ };
874
+ const useGraphQLClient = () => {
875
+ const context = useContext(GraphQLContext);
876
+ if (!context) throw new Error("useGraphQLClient must be used within a GraphQLProvider");
877
+ return context;
878
+ };
879
+
880
+ //#endregion
881
+ //#region src/hooks/useGraphQLConfig.ts
882
+ const useColorsAndFrontendConfig = () => {
883
+ const { getColorsAndFrontendConfig, isReady } = useGraphQLClient();
884
+ const [data, setData] = useState({});
885
+ const [loading, setLoading] = useState(false);
886
+ const [error, setError] = useState(null);
887
+ const fetchConfig = useCallback(async () => {
888
+ if (!isReady) return;
889
+ setLoading(true);
890
+ setError(null);
891
+ try {
892
+ const result = await getColorsAndFrontendConfig();
893
+ setData(result);
894
+ } catch (err) {
895
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown error"));
896
+ } finally {
897
+ setLoading(false);
898
+ }
899
+ }, [getColorsAndFrontendConfig, isReady]);
900
+ useEffect(() => {
901
+ fetchConfig();
902
+ }, [fetchConfig]);
903
+ return {
904
+ data,
905
+ loading,
906
+ error,
907
+ refetch: fetchConfig
908
+ };
909
+ };
910
+ const useOrgId = () => {
911
+ const { getOrgId, isReady } = useGraphQLClient();
912
+ const [orgId, setOrgId] = useState();
913
+ const [loading, setLoading] = useState(false);
914
+ const [error, setError] = useState(null);
915
+ useEffect(() => {
916
+ if (!isReady) return;
917
+ const fetchOrgId = async () => {
918
+ setLoading(true);
919
+ setError(null);
920
+ try {
921
+ const id = await getOrgId();
922
+ setOrgId(id);
923
+ } catch (err) {
924
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown error"));
925
+ } finally {
926
+ setLoading(false);
927
+ }
928
+ };
929
+ fetchOrgId();
930
+ }, [getOrgId, isReady]);
931
+ return {
932
+ orgId,
933
+ loading,
934
+ error
935
+ };
936
+ };
937
+
938
+ //#endregion
939
+ //#region src/contexts/userIdentityContext.tsx
940
+ const getUserAgentDetails = () => {
941
+ const result = new UAParser().getResult();
942
+ return {
943
+ os: result?.os?.name,
944
+ osVersion: result?.os?.version,
945
+ deviceBrand: result?.device?.vendor,
946
+ deviceManufacturer: result?.device?.vendor,
947
+ deviceModel: result?.device?.model,
948
+ browser: result?.browser?.name,
949
+ browserVersion: result?.browser?.version,
950
+ userAgent: result?.ua
951
+ };
952
+ };
953
+ const UserIdentityContext = createContext(void 0);
954
+ const UserIdentityProvider = ({ children }) => {
955
+ const { getItem, setItem, isAvailable: localStorageIsReady } = useLocalStorage();
956
+ const [isReady, setIsReady] = useState(false);
957
+ useEffect(() => {
958
+ setIsReady(localStorageIsReady);
959
+ }, [localStorageIsReady]);
960
+ const USER_ID_OVERRIDE_KEY = "v1-spiffy-user-id-override";
961
+ const USER_ID_DEFAULT_KEY = "v1-spiffy-user-id-default";
962
+ const getUserIdOverrideFromLocalStorage = useCallback(() => {
963
+ return getItem(USER_ID_OVERRIDE_KEY) ?? void 0;
964
+ }, [getItem]);
965
+ const getUserIdDefaultFromLocalStorage = useCallback(() => {
966
+ return getItem(USER_ID_DEFAULT_KEY) ?? void 0;
967
+ }, [getItem]);
968
+ const setUserIdDefaultInLocalStorage = useCallback((userId) => {
969
+ logger_default.logInfo(`setUserIdDefaultInLocalStorage - Setting user_id=${userId}`);
970
+ setItem(USER_ID_DEFAULT_KEY, userId);
971
+ return userId;
972
+ }, [setItem, USER_ID_DEFAULT_KEY]);
973
+ const setUserIdOverrideInLocalStorage = useCallback((userId) => {
974
+ logger_default.logInfo(`setUserIdOverrideInLocalStorage - Setting user_id=${userId}`);
975
+ setItem(USER_ID_OVERRIDE_KEY, userId);
976
+ return userId;
977
+ }, [setItem, USER_ID_OVERRIDE_KEY]);
978
+ const clearUserIdOverrideInLocalStorage = useCallback(() => {
979
+ logger_default.logInfo(`clearUserIdOverrideInLocalStorage - Clearing user_id`);
980
+ setItem(USER_ID_OVERRIDE_KEY, "");
981
+ }, [setItem, USER_ID_OVERRIDE_KEY]);
982
+ const getUserIdOrDefault = useCallback(() => {
983
+ const userIdOverride = getUserIdOverrideFromLocalStorage();
984
+ if (userIdOverride) return userIdOverride;
985
+ const defaultUserId = getUserIdDefaultFromLocalStorage();
986
+ if (defaultUserId) return defaultUserId;
987
+ return setUserIdDefaultInLocalStorage(`spiffy-user-id-${v4()}`);
988
+ }, [
989
+ getUserIdOverrideFromLocalStorage,
990
+ getUserIdDefaultFromLocalStorage,
991
+ setUserIdDefaultInLocalStorage
992
+ ]);
993
+ const identifyUser = useCallback(async () => {
994
+ if (!isReady) {
995
+ logger_default.logWarn("[UserIdentityContext] Context not ready, skipping identifyUser", void 0);
996
+ return;
997
+ }
998
+ try {
999
+ const cdpUserId = "UNKNOWN_CDP_USER_ID";
1000
+ const userId = getUserIdOrDefault();
1001
+ const userAgentDetails = getUserAgentDetails();
1002
+ await commerce_api_default.identifyUser(userId, cdpUserId, userAgentDetails);
1003
+ } catch (error) {
1004
+ logger_default.logError("[spiffy-ai] Error identifying user", error);
1005
+ }
1006
+ }, [isReady, getUserIdOrDefault]);
1007
+ const value = useMemo(() => ({
1008
+ identifyUser,
1009
+ getUserIdOrDefault,
1010
+ getUserIdOverrideFromLocalStorage,
1011
+ getUserIdDefaultFromLocalStorage,
1012
+ setUserIdDefaultInLocalStorage,
1013
+ setUserIdOverrideInLocalStorage,
1014
+ clearUserIdOverrideInLocalStorage,
1015
+ isReady
1016
+ }), [
1017
+ identifyUser,
1018
+ getUserIdOrDefault,
1019
+ getUserIdOverrideFromLocalStorage,
1020
+ getUserIdDefaultFromLocalStorage,
1021
+ setUserIdDefaultInLocalStorage,
1022
+ setUserIdOverrideInLocalStorage,
1023
+ clearUserIdOverrideInLocalStorage,
1024
+ isReady
1025
+ ]);
1026
+ return /* @__PURE__ */ jsx(UserIdentityContext.Provider, {
1027
+ value,
1028
+ children
1029
+ });
1030
+ };
1031
+ const useUserIdentity = () => {
1032
+ const context = useContext(UserIdentityContext);
1033
+ if (!context) throw new Error("useUserIdentity must be used within a UserIdentityProvider");
1034
+ return context;
1035
+ };
1036
+
1037
+ //#endregion
1038
+ //#region src/hooks/useIdentifyUser.ts
1039
+ const useIdentifyUser = () => {
1040
+ const { identifyUser, isReady } = useUserIdentity();
1041
+ const [loading, setLoading] = useState(false);
1042
+ const [error, setError] = useState(null);
1043
+ const executeIdentifyUser = useCallback(async () => {
1044
+ if (!isReady) {
1045
+ setError(/* @__PURE__ */ new Error("UserIdentityContext not ready."));
1046
+ return;
1047
+ }
1048
+ setLoading(true);
1049
+ setError(null);
1050
+ try {
1051
+ await identifyUser();
1052
+ } catch (err) {
1053
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown error during user identification."));
1054
+ throw err;
1055
+ } finally {
1056
+ setLoading(false);
1057
+ }
1058
+ }, [identifyUser, isReady]);
1059
+ return {
1060
+ loading,
1061
+ error,
1062
+ executeIdentifyUser,
1063
+ isReady
1064
+ };
1065
+ };
1066
+
1067
+ //#endregion
1068
+ //#region src/hooks/useImageResolver.ts
1069
+ var ImageResolver = class {};
1070
+ var MerchantImageResolver = class {
1071
+ static {
1072
+ this.imageResolverMap = /* @__PURE__ */ new Map();
1073
+ }
1074
+ static loadMapping() {
1075
+ if (this.imageResolverMap.size === 0) {
1076
+ this.imageResolverMap.set(OrgShortName.Spanx, new ShopifyImageResolver());
1077
+ this.imageResolverMap.set(OrgShortName.SpanxStaging, new ShopifyImageResolver());
1078
+ this.imageResolverMap.set(OrgShortName.UniqueVintage, new ShopifyImageResolver());
1079
+ }
1080
+ return this.imageResolverMap;
1081
+ }
1082
+ static get(name) {
1083
+ return this.loadMapping().get(name);
1084
+ }
1085
+ };
1086
+ var ShopifyImageResolver = class extends ImageResolver {
1087
+ resolve(url, size) {
1088
+ const pattern = /_\d+x\.jpg/;
1089
+ const urlHasPrefix = pattern.test(url);
1090
+ const newSizePrefix = `_${size}x.jpg`;
1091
+ if (urlHasPrefix) return url.replace(pattern, newSizePrefix);
1092
+ return url.replace(".jpg", newSizePrefix);
1093
+ }
1094
+ };
1095
+ const useImageResolver = () => {
1096
+ const orgShortName = useAtomValue(orgShortNameAtom);
1097
+ const resolve = (image, size) => {
1098
+ if (image && size && orgShortName) return MerchantImageResolver.get(orgShortName)?.resolve(image, size) || image;
1099
+ return image;
1100
+ };
1101
+ return { resolve };
1102
+ };
1103
+
1104
+ //#endregion
1105
+ //#region src/hooks/useIntersection.ts
1106
+ const useIntersection = (element, rootMargin) => {
1107
+ const [isVisible, setIsVisible] = useState(false);
1108
+ useEffect(() => {
1109
+ const current = element?.current;
1110
+ const observer = new IntersectionObserver(([entry]) => {
1111
+ setIsVisible(entry.isIntersecting);
1112
+ }, { rootMargin });
1113
+ if (current) observer?.observe(current);
1114
+ return () => {
1115
+ if (current) observer.unobserve(current);
1116
+ };
1117
+ }, []);
1118
+ return isVisible;
1119
+ };
1120
+
1121
+ //#endregion
1122
+ //#region src/hooks/useIsSmallScreen.ts
1123
+ const useIsSmallScreen = () => {
1124
+ const [isSmall, setIsSmall] = useState(false);
1125
+ useEffect(() => {
1126
+ const mediaQuery = window.matchMedia("(max-width: 479px)");
1127
+ setIsSmall(mediaQuery.matches);
1128
+ const handleResize = (event) => {
1129
+ setIsSmall(event.matches);
1130
+ };
1131
+ mediaQuery.addEventListener("change", handleResize);
1132
+ return () => mediaQuery.removeEventListener("change", handleResize);
1133
+ }, []);
1134
+ return isSmall;
1135
+ };
1136
+
1137
+ //#endregion
1138
+ //#region src/hooks/useLocalStorageOperations.ts
1139
+ const useLocalStorageValue = (key) => {
1140
+ const { getItem, setItem, attachListener, detachListener } = useLocalStorage();
1141
+ const [value, setValue] = useState(() => getItem(key));
1142
+ useEffect(() => {
1143
+ const listener = {
1144
+ storageKey: key,
1145
+ listener: (event) => {
1146
+ setValue(event.newValue);
1147
+ }
1148
+ };
1149
+ attachListener(listener);
1150
+ return () => detachListener(listener);
1151
+ }, [
1152
+ key,
1153
+ attachListener,
1154
+ detachListener
1155
+ ]);
1156
+ const updateValue = useCallback((newValue) => {
1157
+ setItem(key, newValue);
1158
+ setValue(newValue);
1159
+ }, [key, setItem]);
1160
+ return {
1161
+ value,
1162
+ setValue: updateValue
1163
+ };
1164
+ };
1165
+ const useSpiffyFeatureFlag = () => {
1166
+ const { setSpiffyOnFeatureFlag } = useLocalStorage();
1167
+ const { value } = useLocalStorageValue(LocalStorageKeys.SpiffyOnOverride);
1168
+ const setFlag = useCallback((flag) => {
1169
+ setSpiffyOnFeatureFlag(flag);
1170
+ }, [setSpiffyOnFeatureFlag]);
1171
+ return {
1172
+ value: value === "true" ? true : value === "false" ? false : null,
1173
+ setFlag
1174
+ };
1175
+ };
1176
+ const useEnviveFeatureFlag = () => {
1177
+ const { setItem, getItem } = useLocalStorage();
1178
+ const { value } = useLocalStorageValue(LocalStorageKeys.EnviveOnOverride);
1179
+ const setFlag = useCallback((flag) => {
1180
+ if (flag === true) setItem(LocalStorageKeys.EnviveOnOverride, "true");
1181
+ else if (flag === false) setItem(LocalStorageKeys.EnviveOnOverride, "false");
1182
+ }, [setItem]);
1183
+ return {
1184
+ value: value === "true" ? true : value === "false" ? false : null,
1185
+ setFlag
1186
+ };
1187
+ };
1188
+ const useLocalStorageListener = (key, callback) => {
1189
+ const { attachListener, detachListener } = useLocalStorage();
1190
+ useEffect(() => {
1191
+ const listener = {
1192
+ storageKey: key,
1193
+ listener: callback
1194
+ };
1195
+ attachListener(listener);
1196
+ return () => detachListener(listener);
1197
+ }, [
1198
+ key,
1199
+ callback,
1200
+ attachListener,
1201
+ detachListener
1202
+ ]);
1203
+ };
1204
+
1205
+ //#endregion
1206
+ //#region src/hooks/useMessageFilter.ts
1207
+ const useMessageFilter = () => {
1208
+ const findMessageIndex = ({ msgs, type, role }) => {
1209
+ let lastIndex = -1;
1210
+ msgs.forEach((subArray, index) => {
1211
+ subArray.forEach((obj) => {
1212
+ if (obj.type === type || obj.role === role) lastIndex = index;
1213
+ });
1214
+ });
1215
+ return lastIndex;
1216
+ };
1217
+ const removePreviousDiscussions = (msgs, index) => {
1218
+ if (index > -1) {
1219
+ const lastMessages = msgs.slice(index);
1220
+ return lastMessages.length > 0 ? lastMessages : msgs;
1221
+ }
1222
+ return msgs;
1223
+ };
1224
+ const getFilteredMessages = (msgs, skipFilter) => {
1225
+ const messageMap = msgs.reduce((acc, msg) => {
1226
+ acc[msg[0].id] = msg;
1227
+ return acc;
1228
+ }, {});
1229
+ const deduplicatedMsgs = Object.values(messageMap);
1230
+ if (!skipFilter) {
1231
+ const idx = findMessageIndex({
1232
+ msgs: deduplicatedMsgs,
1233
+ type: MessageType.Separator
1234
+ });
1235
+ return removePreviousDiscussions(deduplicatedMsgs, idx);
1236
+ }
1237
+ return deduplicatedMsgs;
1238
+ };
1239
+ return {
1240
+ findMessageIndex,
1241
+ removePreviousDiscussions,
1242
+ getFilteredMessages
1243
+ };
1244
+ };
1245
+
1246
+ //#endregion
1247
+ //#region src/hooks/useMessageScrollObserver.ts
1248
+ const useMessageScrollObserver = (boxRef, scrollRef, onScrollChange) => {
1249
+ const calculateScrollHeight = () => {
1250
+ const boxHeight = boxRef?.current?.getBoundingClientRect().height || 0;
1251
+ const scrollHeight = scrollRef?.current?.getBoundingClientRect().height || 0;
1252
+ return boxHeight - scrollHeight;
1253
+ };
1254
+ const updateState = () => {
1255
+ const scrollHeight = calculateScrollHeight();
1256
+ if (scrollHeight > 0) onScrollChange(scrollHeight);
1257
+ };
1258
+ useEffect(() => {
1259
+ let boxRO = null;
1260
+ let scrollRO = null;
1261
+ if (scrollRef?.current) {
1262
+ boxRO = new ResizeObserver(updateState);
1263
+ boxRO.observe(scrollRef?.current);
1264
+ }
1265
+ if (boxRef?.current) {
1266
+ scrollRO = new ResizeObserver(updateState);
1267
+ scrollRO.observe(boxRef?.current);
1268
+ }
1269
+ return () => {
1270
+ if (scrollRef?.current && boxRO) boxRO.unobserve(scrollRef?.current);
1271
+ if (scrollRO && boxRef?.current) scrollRO?.unobserve(boxRef?.current);
1272
+ };
1273
+ }, []);
1274
+ };
1275
+
1276
+ //#endregion
1277
+ //#region src/contexts/featureFlagServiceContext.tsx
1278
+ var FeatureFlagService = class {
1279
+ constructor(featureGates) {
1280
+ this.isFeatureGateEnabled = (featureGate) => {
1281
+ const gateValue = this.featureGates.find((gate) => gate.name === featureGate);
1282
+ if (gateValue == null || gateValue.value == null) {
1283
+ logger_default.logDebug(`[spiffy-ai] isFeatureGateEnabled featureGate:${featureGate} value is undefined - returning false`);
1284
+ return false;
1285
+ }
1286
+ return gateValue.value;
1287
+ };
1288
+ this.isClientSessionEnabled = () => {
1289
+ return this.featureGates.filter((gate) => gate.name === FeatureGates.IsClientSessionEnabled && gate.value === true).length > 0;
1290
+ };
1291
+ this.getFeatureFlags = () => {
1292
+ return Object.fromEntries(Object.values(FeatureGates).map((featureGate) => [featureGate, this.isFeatureGateEnabled(featureGate)]));
1293
+ };
1294
+ this.featureGates = featureGates;
1295
+ }
1296
+ };
1297
+ const FeatureFlagServiceContext = createContext(void 0);
1298
+ const FeatureFlagServiceProvider = ({ featureGates, children }) => {
1299
+ const featureFlagService = useMemo(() => new FeatureFlagService(featureGates), [featureGates]);
1300
+ return /* @__PURE__ */ jsx(FeatureFlagServiceContext.Provider, {
1301
+ value: { featureFlagService },
1302
+ children
1303
+ });
1304
+ };
1305
+ const useFeatureFlagService = () => {
1306
+ const context = useContext(FeatureFlagServiceContext);
1307
+ if (context === void 0) throw new Error("useFeatureFlagService must be used within a FeatureFlagServiceProvider");
1308
+ return context;
1309
+ };
1310
+
1311
+ //#endregion
1312
+ //#region src/contexts/newOrgConfigContext.tsx
1313
+ const NewOrgConfigContext = createContext(void 0);
1314
+ const NewOrgConfigProvider = ({ children }) => {
1315
+ const [oldConfig, setOldConfig] = useState();
1316
+ const orgShortName = useAtomValue(orgShortNameAtom);
1317
+ const setNewOrgConfig = useSetAtom(newOrgConfigAtom);
1318
+ const { data: newConfig, loading, error } = useColorsAndFrontendConfig();
1319
+ useEffect(() => {
1320
+ if (orgShortName) getOrgInfo(orgShortName).then(setOldConfig);
1321
+ }, [orgShortName]);
1322
+ const combinedConfig = useMemo(() => {
1323
+ if (!oldConfig || !newConfig) return null;
1324
+ return {
1325
+ ...oldConfig,
1326
+ ...newConfig
1327
+ };
1328
+ }, [oldConfig, newConfig]);
1329
+ useEffect(() => {
1330
+ const atomStore = getAtomStore();
1331
+ if (combinedConfig) {
1332
+ atomStore.set(orgIdAtom, "mock-org-id");
1333
+ setNewOrgConfig(combinedConfig);
1334
+ }
1335
+ }, [combinedConfig, setNewOrgConfig]);
1336
+ const contextValue = useMemo(() => {
1337
+ if (!orgShortName || loading && !oldConfig) return {
1338
+ combinedConfig: null,
1339
+ loading: true,
1340
+ error: null
1341
+ };
1342
+ if (error) return {
1343
+ combinedConfig: null,
1344
+ loading: false,
1345
+ error
1346
+ };
1347
+ return {
1348
+ combinedConfig,
1349
+ loading: false,
1350
+ error: null
1351
+ };
1352
+ }, [
1353
+ orgShortName,
1354
+ loading,
1355
+ error,
1356
+ oldConfig,
1357
+ combinedConfig
1358
+ ]);
1359
+ return /* @__PURE__ */ jsx(NewOrgConfigContext.Provider, {
1360
+ value: contextValue,
1361
+ children: /* @__PURE__ */ jsx(FeatureFlagServiceProvider, {
1362
+ featureGates: [],
1363
+ children
1364
+ })
1365
+ });
1366
+ };
1367
+ const useNewOrgConfigContext = () => {
1368
+ const context = useContext(NewOrgConfigContext);
1369
+ if (context === void 0) throw new Error("useNewOrgConfigContext must be used within a NewOrgConfigProvider");
1370
+ return context;
1371
+ };
1372
+
1373
+ //#endregion
1374
+ //#region src/hooks/useNewOrgConfig.ts
1375
+ const useNewOrgConfig = () => {
1376
+ const { combinedConfig, loading, error } = useNewOrgConfigContext();
1377
+ return {
1378
+ ...combinedConfig,
1379
+ loading,
1380
+ error
1381
+ };
1382
+ };
1383
+
1384
+ //#endregion
1385
+ //#region src/hooks/utils.ts
1386
+ const isElementPartiallyVisible = (el) => {
1387
+ if (!el) return false;
1388
+ const rect = el.getBoundingClientRect();
1389
+ const windowHeight = window.innerHeight || document.documentElement.clientHeight;
1390
+ const windowWidth = window.innerWidth || document.documentElement.clientWidth;
1391
+ const verticallyVisible = Math.round(rect.top) < windowHeight && Math.round(rect.bottom) > 0;
1392
+ const horizontallyVisible = Math.round(rect.left) < windowWidth && Math.round(rect.right) > 0;
1393
+ return verticallyVisible && horizontallyVisible;
1394
+ };
1395
+ const createAppLoadedEvent = () => ({
1396
+ eventId: v4(),
1397
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1398
+ category: UserEventCategory.AppLoaded
1399
+ });
1400
+ const createVisitUserEvent = ({ variantInfo }) => {
1401
+ if (variantInfo.variant === "pdp" && variantInfo.productId != null) return {
1402
+ eventId: v4(),
1403
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1404
+ category: UserEventCategory.PdpVisit,
1405
+ attributes: {
1406
+ productId: variantInfo.productId,
1407
+ parentProductId: variantInfo.parentProductId ?? "",
1408
+ url: variantInfo.url ?? ""
1409
+ }
1410
+ };
1411
+ if (variantInfo.variant === "plp" && variantInfo.plpId != null) return {
1412
+ eventId: v4(),
1413
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1414
+ category: UserEventCategory.PlpVisit,
1415
+ attributes: {
1416
+ category: PLPAttributeCategory.Id,
1417
+ attributes: { id: variantInfo.plpId }
1418
+ }
1419
+ };
1420
+ if (variantInfo.variant === "page_visit") return {
1421
+ eventId: v4(),
1422
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1423
+ category: UserEventCategory.PageVisit,
1424
+ attributes: {
1425
+ url: variantInfo.url,
1426
+ pageVisitCategory: variantInfo.pageVisitCategory
1427
+ }
1428
+ };
1429
+ };
1430
+ const parseTime = (time, timeZone) => {
1431
+ const times = time.match(/^([0-1]?\d):([0-5]\d)(AM|PM)$/i);
1432
+ const hours = times?.[1];
1433
+ const minutes = times?.[2];
1434
+ const period = times?.[3];
1435
+ if (hours && minutes && period) {
1436
+ const date = /* @__PURE__ */ new Date();
1437
+ let adjustedHours = 0;
1438
+ if (period.toUpperCase() === "PM" && hours !== "12") adjustedHours = parseInt(hours) + 12;
1439
+ if (period.toUpperCase() === "AM" && hours !== "12") adjustedHours = parseInt(hours);
1440
+ const formattedDate = `${date.toISOString().split("T")[0]}T${String(adjustedHours).padStart(2, "0")}:${minutes}:00`;
1441
+ return /* @__PURE__ */ new Date(`${formattedDate}${timeZone}`);
1442
+ }
1443
+ };
1444
+ const isWithinBusinessHours = (startTime, endTime, timeZone) => {
1445
+ const start = parseTime(startTime, timeZone);
1446
+ let end = parseTime(endTime, timeZone);
1447
+ if (!start || !end) return false;
1448
+ let now = /* @__PURE__ */ new Date();
1449
+ if (end < start) {
1450
+ end = new Date(end.getTime() + 1440 * 60 * 1e3);
1451
+ now = new Date(now.getTime() + 1440 * 60 * 1e3);
1452
+ }
1453
+ if (end.getUTCDate() > start.getUTCDate()) {
1454
+ const crossingMidnight = new Date(now.getTime() + 1440 * 60 * 1e3);
1455
+ return start <= now || crossingMidnight <= end;
1456
+ }
1457
+ return now >= start && now <= end;
1458
+ };
1459
+ let SearchResultsState = /* @__PURE__ */ function(SearchResultsState$1) {
1460
+ SearchResultsState$1[SearchResultsState$1["Loading"] = 0] = "Loading";
1461
+ SearchResultsState$1[SearchResultsState$1["Results"] = 1] = "Results";
1462
+ SearchResultsState$1[SearchResultsState$1["NoResults"] = 2] = "NoResults";
1463
+ return SearchResultsState$1;
1464
+ }({});
1465
+ const getSearchResultsState = (isLoadingSearch, searchData) => {
1466
+ if (isLoadingSearch) return SearchResultsState.Loading;
1467
+ if (searchData) return SearchResultsState.Results;
1468
+ return SearchResultsState.NoResults;
1469
+ };
1470
+
1471
+ //#endregion
1472
+ //#region src/hooks/useSearch.tsx
1473
+ const useSearch = () => {
1474
+ const config = useNewOrgConfig();
1475
+ const orgShortName = useAtomValue(orgShortNameAtom);
1476
+ const { data: searchData, loading: isLoadingSearch } = useAtomValue(searchAtom);
1477
+ const productList = useAtomValue(filteredSearchProductsAtom);
1478
+ const performSearch = useSetAtom(performSearchAtom);
1479
+ const [{ results: autocompleteResults, isLoading: isLoadingAutocomplete }, setAutocompleteState] = useAtom(autocompleteStateAtom);
1480
+ const [{ query }] = useAtom(searchParamsAtom);
1481
+ const [isFilterOpen, setIsFilterOpen] = useAtom(isFilterOpenAtom);
1482
+ const [selectedFilterOptions] = useAtom(searchSelectedFiltersAtom);
1483
+ const addFilter = useSetAtom(addSearchFilterAtom);
1484
+ const removeFilter = useSetAtom(removeSearchFilterAtom);
1485
+ const [productSorting, setProductSorting] = useAtom(searchProductSortingAtom);
1486
+ const clearFilters = useSetAtom(clearSearchFiltersAtom);
1487
+ const searchFilters = useAtomValue(searchFiltersAtom);
1488
+ const [isDirty, setIsDirty] = useState(true);
1489
+ const [focusedIndex, setFocusedIndex] = useState(-1);
1490
+ const [focusedOptionId, setFocusedOptionId] = useState(void 0);
1491
+ const [searchText, setSearchText] = useState(query || "");
1492
+ const searchResultsRef = useRef(null);
1493
+ const debouncedSearchText = useDebounce(searchText, 200);
1494
+ const searchResultsState = getSearchResultsState(isLoadingSearch, searchData);
1495
+ const dynamicFilters = searchData?.filters || [];
1496
+ const safeProductCardConfig = config?.frontendConfig?.uiConfigs?.productCardConfig || {
1497
+ variant: "minimal",
1498
+ hoverVariant: "none",
1499
+ layoutVariant: "square"
1500
+ };
1501
+ const safeMerchantShortName = orgShortName || "";
1502
+ const availableDynamicFilters = useMemo(() => {
1503
+ return dynamicFilters.filter((dynamicFilterName) => !selectedFilterOptions.some((option) => option.id === `dynamic:${dynamicFilterName}`)).map((dynamicFilterName) => ({
1504
+ name: dynamicFilterName,
1505
+ displayName: formatFilterDisplayName(dynamicFilterName)
1506
+ }));
1507
+ }, [dynamicFilters, selectedFilterOptions]);
1508
+ const filters = useMemo(() => {
1509
+ return [{
1510
+ filterId: "sort",
1511
+ displayName: "SORT",
1512
+ items: [
1513
+ {
1514
+ filterItemId: String(ProductSorting.FEATURED),
1515
+ displayName: "Relevance",
1516
+ productCount: 0,
1517
+ isSelected: productSorting === ProductSorting.FEATURED
1518
+ },
1519
+ {
1520
+ filterItemId: String(ProductSorting.PRICE_ASC),
1521
+ displayName: "Price: Low to High",
1522
+ productCount: 0,
1523
+ isSelected: productSorting === ProductSorting.PRICE_ASC
1524
+ },
1525
+ {
1526
+ filterItemId: String(ProductSorting.PRICE_DESC),
1527
+ displayName: "Price: High to Low",
1528
+ productCount: 0,
1529
+ isSelected: productSorting === ProductSorting.PRICE_DESC
1530
+ }
1531
+ ]
1532
+ }, ...searchFilters];
1533
+ }, [productSorting, searchFilters]);
1534
+ const filterButtonText = useMemo(() => {
1535
+ const selectedCount = filters.reduce((acc, filter) => {
1536
+ if (filter.filterId === "sort") return acc;
1537
+ return acc + filter.items.filter((item) => item.isSelected).length;
1538
+ }, 0);
1539
+ if (selectedCount === 0) return "Filter & Sort";
1540
+ return `Filter & Sort (${selectedCount})`;
1541
+ }, [filters]);
1542
+ const { trackEvent } = useAmplitude();
1543
+ const handleToggleDynamicFilter = useCallback(({ filter, dynamicFilterDisplayName }) => {
1544
+ trackEvent({
1545
+ eventName: SpiffyMetricsEventName.SearchFilterClicked,
1546
+ eventProps: {
1547
+ filterType: "Dynamic",
1548
+ filterValue: filter,
1549
+ queryText: searchText
1550
+ }
1551
+ });
1552
+ addFilter(createFilterOption("dynamic", filter, dynamicFilterDisplayName));
1553
+ }, [
1554
+ addFilter,
1555
+ searchText,
1556
+ trackEvent
1557
+ ]);
1558
+ const handleRemoveFilter = useCallback((filter) => {
1559
+ removeFilter(filter.id);
1560
+ }, [removeFilter]);
1561
+ const handleSubmitSearch = useCallback(async () => {
1562
+ if (searchText.trim()) {
1563
+ trackEvent({
1564
+ eventName: SpiffyMetricsEventName.SearchQuerySubmitted,
1565
+ eventProps: {
1566
+ searchOrigin: SpiffyWidgets.SearchResults,
1567
+ queryText: searchText.trim()
1568
+ },
1569
+ alsoSendToGoogleAnalytics: true
1570
+ });
1571
+ const url = new URL(window.location.href);
1572
+ url.searchParams.set("esq", searchText.trim());
1573
+ window.history.pushState({}, "", url);
1574
+ performSearch({ query: searchText.trim() });
1575
+ }
1576
+ }, [
1577
+ performSearch,
1578
+ searchText,
1579
+ trackEvent
1580
+ ]);
1581
+ const handleAutocompleteSelect = useCallback((suggestion) => {
1582
+ setSearchText(suggestion);
1583
+ handleSubmitSearch();
1584
+ }, [handleSubmitSearch, setSearchText]);
1585
+ const handleKeyDown = useCallback((event) => {
1586
+ if (event.key === "ArrowDown") {
1587
+ event.preventDefault();
1588
+ const newIndex = (focusedIndex + 1) % autocompleteResults.length;
1589
+ setFocusedIndex(newIndex);
1590
+ setFocusedOptionId(`option-${newIndex}`);
1591
+ } else if (event.key === "ArrowUp") {
1592
+ event.preventDefault();
1593
+ const newIndex = (focusedIndex - 1 + autocompleteResults.length) % autocompleteResults.length;
1594
+ setFocusedIndex(newIndex);
1595
+ setFocusedOptionId(`option-${newIndex}`);
1596
+ } else if (event.key === "Enter") if (focusedIndex === -1) {
1597
+ event.preventDefault();
1598
+ handleSubmitSearch();
1599
+ } else {
1600
+ event.preventDefault();
1601
+ const suggestionText = autocompleteResults[focusedIndex];
1602
+ handleAutocompleteSelect(suggestionText);
1603
+ }
1604
+ else if (event.key === "Escape") {
1605
+ event.preventDefault();
1606
+ setFocusedIndex(-1);
1607
+ setFocusedOptionId(void 0);
1608
+ }
1609
+ }, [
1610
+ autocompleteResults,
1611
+ focusedIndex,
1612
+ handleAutocompleteSelect,
1613
+ handleSubmitSearch
1614
+ ]);
1615
+ const handleSearchInputChange = (newValue) => {
1616
+ if (newValue.length === 1) trackEvent({
1617
+ eventName: SpiffyMetricsEventName.SearchInputStarted,
1618
+ eventProps: { searchOrigin: SpiffyWidgets.SearchResults }
1619
+ });
1620
+ setSearchText(newValue);
1621
+ setIsDirty(true);
1622
+ };
1623
+ const handleSelectFilterItem = useCallback(({ filterId, filterItemId, isSelected, displayName }) => {
1624
+ if (filterId === "sort") {
1625
+ const newSort = filterItemId;
1626
+ trackEvent({
1627
+ eventName: SpiffyMetricsEventName.SearchSortClicked,
1628
+ eventProps: {
1629
+ sortType: newSort,
1630
+ queryText: searchText
1631
+ }
1632
+ });
1633
+ setProductSorting(newSort);
1634
+ } else if (!isSelected) removeFilter(`${filterId}:${filterItemId}`);
1635
+ else {
1636
+ trackEvent({
1637
+ eventName: SpiffyMetricsEventName.SearchFilterClicked,
1638
+ eventProps: {
1639
+ filterType: "Static",
1640
+ filterCategory: filterId,
1641
+ filterValue: filterItemId,
1642
+ queryText: searchText
1643
+ }
1644
+ });
1645
+ addFilter(createFilterOption(filterId, filterItemId, displayName));
1646
+ }
1647
+ }, [
1648
+ addFilter,
1649
+ removeFilter,
1650
+ setProductSorting,
1651
+ searchText,
1652
+ trackEvent
1653
+ ]);
1654
+ const handleClearAllFilters = useCallback(() => {
1655
+ setProductSorting(ProductSorting.FEATURED);
1656
+ clearFilters();
1657
+ }, [setProductSorting, clearFilters]);
1658
+ useTrackComponentVisibleEvent(SpiffyWidgets.SearchResults, searchResultsRef, {}, SpiffyMetricsEventName.SearchComponentVisible);
1659
+ useEffect(() => {
1660
+ if (productList.length > 0) trackEvent({
1661
+ eventName: SpiffyMetricsEventName.SearchResultsViewed,
1662
+ eventProps: {
1663
+ queryText: searchText,
1664
+ resultsCount: productList.length
1665
+ }
1666
+ });
1667
+ }, [
1668
+ productList.length,
1669
+ searchText,
1670
+ trackEvent
1671
+ ]);
1672
+ useEffect(() => {
1673
+ if (query && query !== searchText) setSearchText(query);
1674
+ }, [query]);
1675
+ useEffect(() => {
1676
+ const esq = new URLSearchParams(window.location.search).get("esq");
1677
+ if (esq) {
1678
+ setSearchText(esq);
1679
+ performSearch({ query: esq });
1680
+ }
1681
+ }, [performSearch]);
1682
+ const fetchAutocompleteSuggestions = (_query) => {
1683
+ return Promise.resolve([]);
1684
+ };
1685
+ useEffect(() => {
1686
+ if (fetchAutocompleteSuggestions === void 0) return;
1687
+ if (!isDirty || debouncedSearchText.length <= 2) {
1688
+ setAutocompleteState({
1689
+ results: [],
1690
+ isLoading: false
1691
+ });
1692
+ return;
1693
+ }
1694
+ setAutocompleteState((prev) => ({
1695
+ ...prev,
1696
+ isLoading: true
1697
+ }));
1698
+ const fetchData = async () => {
1699
+ try {
1700
+ const results = await fetchAutocompleteSuggestions?.(debouncedSearchText);
1701
+ setAutocompleteState({
1702
+ results: results ?? [],
1703
+ isLoading: false
1704
+ });
1705
+ } catch (error) {
1706
+ logger_default.logError("Failed to fetch autocomplete suggestions:", error);
1707
+ setAutocompleteState({
1708
+ results: [],
1709
+ isLoading: false
1710
+ });
1711
+ }
1712
+ };
1713
+ fetchData();
1714
+ }, [
1715
+ debouncedSearchText,
1716
+ isDirty,
1717
+ setAutocompleteState
1718
+ ]);
1719
+ return {
1720
+ searchData,
1721
+ searchResponseId: searchData?.searchResponseId ?? "",
1722
+ merchantShortName: safeMerchantShortName,
1723
+ productCardConfig: safeProductCardConfig,
1724
+ productList,
1725
+ autocompleteResults,
1726
+ searchFilters: filters,
1727
+ availableDynamicFilters,
1728
+ selectedFilterOptions,
1729
+ searchText,
1730
+ searchResultsState,
1731
+ isLoadingAutocomplete,
1732
+ isLoadingSearch,
1733
+ isFilterOpen,
1734
+ isDirty,
1735
+ focusedIndex,
1736
+ focusedOptionId,
1737
+ filterButtonText,
1738
+ onSearchInputChange: handleSearchInputChange,
1739
+ onSubmitSearch: handleSubmitSearch,
1740
+ onAutocompleteSelect: handleAutocompleteSelect,
1741
+ onKeyDown: handleKeyDown,
1742
+ onToggleDynamicFilter: handleToggleDynamicFilter,
1743
+ onSelectFilterItem: handleSelectFilterItem,
1744
+ onRemoveFilter: handleRemoveFilter,
1745
+ onClearAllFilters: handleClearAllFilters,
1746
+ setIsFilterOpen,
1747
+ searchResultsRef
1748
+ };
1749
+ };
1750
+
1751
+ //#endregion
1752
+ //#region src/hooks/useAppDetails.ts
1753
+ const useAppDetails = () => {
1754
+ const { orgId: fetchedOrgId } = useOrgId();
1755
+ const orgId = fetchedOrgId ?? "";
1756
+ const orgShortName = useAtomValue(orgShortNameAtom) ?? "spiffy-ai";
1757
+ const chatId = useAtomValue(chatIdAtom);
1758
+ const userId = useAtomValue(userIdAtom);
1759
+ const source = useAtomValue(contextSourceAtom) ?? ContextSourceEnum.App;
1760
+ const env = useAtomValue(envAtom) ?? ContextEnvEnum.Dev;
1761
+ const variantInfo = useAtomValue(variantInfoAtom);
1762
+ return {
1763
+ orgId,
1764
+ orgShortName,
1765
+ chatId,
1766
+ userId,
1767
+ source,
1768
+ env,
1769
+ variantInfo
1770
+ };
1771
+ };
1772
+
1773
+ //#endregion
1774
+ //#region src/contexts/searchContext.tsx
1775
+ const transformProductResponses = (products) => products.map((data) => ({
1776
+ id: data.id,
1777
+ responseId: data.response_id,
1778
+ category: ResponseCategory.Product,
1779
+ description: data.description,
1780
+ imageUrl: data.image_url,
1781
+ imageUrls: data.image_urls,
1782
+ title: data.title,
1783
+ url: data.url,
1784
+ originalPrice: data.original_price,
1785
+ salePrice: data.sale_price,
1786
+ averageRating: data.average_rating,
1787
+ numberReviews: data.number_reviews,
1788
+ metadata: data.metadata,
1789
+ isForGrid: data.is_for_grid,
1790
+ colors: data.colors,
1791
+ sizes: data.sizes,
1792
+ filters: data.filters
1793
+ }));
1794
+ async function errorResponseBody(error) {
1795
+ try {
1796
+ return await error.response.json();
1797
+ } catch {
1798
+ return {};
1799
+ }
1800
+ }
1801
+ async function throwSessionRestartRequiredIf(errorMsg, error) {
1802
+ if (!(error instanceof ResponseError)) {
1803
+ logger_default.logInfo(errorMsg, error);
1804
+ throw error;
1805
+ }
1806
+ const errorResponse = await errorResponseBody(error);
1807
+ if (errorResponse?.message?.toLowerCase() === "unsupported product" || errorResponse?.app_code?.toUpperCase() === "PRODUCT_NOT_FOUND") throw new UnsupportedProductException();
1808
+ else if (errorResponse?.app_code?.toUpperCase() === "RESTART_SESSION" || errorResponse?.sub_code?.toUpperCase() === "NOT_FOUND") {
1809
+ logger_default.logInfo("Session does not exist. Re-start session", error, error.response, errorResponse);
1810
+ throw new SessionRestartRequired();
1811
+ }
1812
+ logger_default.logInfo(errorMsg, error);
1813
+ throw error;
1814
+ }
1815
+ const SearchContext = createContext(void 0);
1816
+ const SearchProvider = ({ children }) => {
1817
+ const { orgLevelApiKey, publicKey } = useEnviveConfig();
1818
+ const apiKey = orgLevelApiKey || publicKey;
1819
+ const appDetails = useAppDetails();
1820
+ const baseUrl = useAtomValue(baseUrlAtom);
1821
+ const isReady = Boolean(apiKey && appDetails && baseUrl);
1822
+ const searchApi = useMemo(() => {
1823
+ if (!isReady) return null;
1824
+ const config = new Configuration({
1825
+ accessToken: apiKey,
1826
+ basePath: baseUrl,
1827
+ headers: {
1828
+ "Content-Type": "application/json",
1829
+ Accept: "application/json"
1830
+ }
1831
+ });
1832
+ return new SearchApi(config);
1833
+ }, [
1834
+ apiKey,
1835
+ baseUrl,
1836
+ isReady
1837
+ ]);
1838
+ const searchProducts = useCallback(async (params) => {
1839
+ if (!isReady || !searchApi) throw new Error("SearchService not ready - missing dependencies");
1840
+ try {
1841
+ const { products, filters, search_response_id: searchResponseId } = await searchApi.v1SearchQueryGet({
1842
+ query: params.query,
1843
+ limit: params.limit,
1844
+ org_id: appDetails.orgId,
1845
+ user_id: appDetails.userId
1846
+ });
1847
+ return {
1848
+ products: transformProductResponses(products) || [],
1849
+ filters: filters || [],
1850
+ totalProductCount: products?.length || 0,
1851
+ searchResponseId: searchResponseId || ""
1852
+ };
1853
+ } catch (error) {
1854
+ await throwSessionRestartRequiredIf("Failed to search products", error);
1855
+ return {
1856
+ products: [],
1857
+ filters: [],
1858
+ totalProductCount: 0,
1859
+ searchResponseId: ""
1860
+ };
1861
+ }
1862
+ }, [
1863
+ searchApi,
1864
+ isReady,
1865
+ appDetails
1866
+ ]);
1867
+ useEffect(() => {
1868
+ if (isReady) setSearchServiceFunction(searchProducts);
1869
+ else clearSearchServiceFunction();
1870
+ return () => {
1871
+ clearSearchServiceFunction();
1872
+ };
1873
+ }, [searchProducts, isReady]);
1874
+ const value = useMemo(() => ({
1875
+ searchProducts,
1876
+ isReady
1877
+ }), [searchProducts, isReady]);
1878
+ return /* @__PURE__ */ jsx(SearchContext.Provider, {
1879
+ value,
1880
+ children
1881
+ });
1882
+ };
1883
+ const useSearchService = () => {
1884
+ const context = useContext(SearchContext);
1885
+ if (!context) throw new Error("useSearchService must be used within a SearchProvider");
1886
+ return context;
1887
+ };
1888
+
1889
+ //#endregion
1890
+ //#region src/hooks/useSearchOperations.ts
1891
+ const useProductSearch = () => {
1892
+ const { searchProducts, isReady } = useSearchService();
1893
+ const [data, setData] = useState();
1894
+ const [loading, setLoading] = useState(false);
1895
+ const [error, setError] = useState(null);
1896
+ const search = useCallback(async (params) => {
1897
+ if (!isReady) {
1898
+ setError(/* @__PURE__ */ new Error("Search service not ready - missing dependencies"));
1899
+ return;
1900
+ }
1901
+ setLoading(true);
1902
+ setError(null);
1903
+ try {
1904
+ const result = await searchProducts(params);
1905
+ setData(result);
1906
+ } catch (err) {
1907
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown search error"));
1908
+ } finally {
1909
+ setLoading(false);
1910
+ }
1911
+ }, [searchProducts, isReady]);
1912
+ const reset = useCallback(() => {
1913
+ setData(void 0);
1914
+ setError(null);
1915
+ setLoading(false);
1916
+ }, []);
1917
+ return {
1918
+ data,
1919
+ loading,
1920
+ error,
1921
+ search,
1922
+ reset,
1923
+ isReady
1924
+ };
1925
+ };
1926
+ const useSearchWithQuery = (params) => {
1927
+ const { searchProducts, isReady } = useSearchService();
1928
+ const [data, setData] = useState();
1929
+ const [loading, setLoading] = useState(false);
1930
+ const [error, setError] = useState(null);
1931
+ const executeSearch = useCallback(async (searchParams) => {
1932
+ if (!isReady) return;
1933
+ setLoading(true);
1934
+ setError(null);
1935
+ try {
1936
+ const result = await searchProducts(searchParams);
1937
+ setData(result);
1938
+ } catch (err) {
1939
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown search error"));
1940
+ } finally {
1941
+ setLoading(false);
1942
+ }
1943
+ }, [searchProducts, isReady]);
1944
+ useEffect(() => {
1945
+ if (params && isReady) executeSearch(params);
1946
+ }, [
1947
+ params,
1948
+ isReady,
1949
+ executeSearch
1950
+ ]);
1951
+ const refetch = useCallback(() => {
1952
+ if (params) executeSearch(params);
1953
+ }, [params, executeSearch]);
1954
+ return {
1955
+ data,
1956
+ loading,
1957
+ error,
1958
+ refetch,
1959
+ isReady
1960
+ };
1961
+ };
1962
+
1963
+ //#endregion
1964
+ //#region src/contexts/sessionStorageContext.tsx
1965
+ const SessionStorageContext = createContext(null);
1966
+ const SessionStorageProvider = ({ children }) => {
1967
+ const isAvailable = useMemo(() => {
1968
+ try {
1969
+ return typeof window !== "undefined" && !!window.sessionStorage;
1970
+ } catch {
1971
+ return false;
1972
+ }
1973
+ }, []);
1974
+ useEffect(() => {
1975
+ if (!isAvailable) logger_default.logError("sessionStorage is not available", void 0);
1976
+ }, [isAvailable]);
1977
+ const setItem = useCallback((key, value$1) => {
1978
+ if (!isAvailable) return;
1979
+ sessionStorage.setItem(key, value$1);
1980
+ window.dispatchEvent(new StorageEvent("storage", {
1981
+ key,
1982
+ newValue: value$1
1983
+ }));
1984
+ }, [isAvailable]);
1985
+ const getItem = useCallback((key) => {
1986
+ if (!isAvailable) return null;
1987
+ return sessionStorage.getItem(key);
1988
+ }, [isAvailable]);
1989
+ const value = useMemo(() => ({
1990
+ setItem,
1991
+ getItem,
1992
+ isAvailable
1993
+ }), [
1994
+ setItem,
1995
+ getItem,
1996
+ isAvailable
1997
+ ]);
1998
+ return /* @__PURE__ */ jsx(SessionStorageContext.Provider, {
1999
+ value,
2000
+ children
2001
+ });
2002
+ };
2003
+ const useSessionStorage = () => {
2004
+ const context = useContext(SessionStorageContext);
2005
+ if (!context) throw new Error("useSessionStorage must be used within a SessionStorageProvider");
2006
+ return context;
2007
+ };
2008
+
2009
+ //#endregion
2010
+ //#region src/hooks/useSessionStorageOperations.ts
2011
+ const useSessionStorageValue = (key) => {
2012
+ const { getItem, setItem } = useSessionStorage();
2013
+ const [value, setValue] = useState(() => getItem(key));
2014
+ useEffect(() => {
2015
+ const handleStorageChange = (event) => {
2016
+ if (event.key === key) setValue(event.newValue);
2017
+ };
2018
+ window.addEventListener("storage", handleStorageChange);
2019
+ return () => window.removeEventListener("storage", handleStorageChange);
2020
+ }, [key, getItem]);
2021
+ const updateValue = useCallback((newValue) => {
2022
+ setItem(key, newValue);
2023
+ setValue(newValue);
2024
+ }, [key, setItem]);
2025
+ return {
2026
+ value,
2027
+ setValue: updateValue
2028
+ };
2029
+ };
2030
+
2031
+ //#endregion
2032
+ //#region src/contexts/shopifyUrlContext.tsx
2033
+ const ShopifyUrlContext = createContext(void 0);
2034
+ const ShopifyUrlProvider = ({ children }) => {
2035
+ const isReady = true;
2036
+ const getTrimmedPathName = useCallback(() => {
2037
+ let { pathname } = window.location;
2038
+ pathname = pathname.replace("/proxy", "");
2039
+ pathname = pathname.replace(/#.*$/, "");
2040
+ pathname = pathname.replace(/\/$/, "");
2041
+ if (pathname === void 0 || pathname === null || pathname.length === 0) return null;
2042
+ return pathname;
2043
+ }, []);
2044
+ const getPlpOrPdpId = useCallback((extractor) => {
2045
+ if (extractor === "shopify-product-variant-id") {
2046
+ const variantId = parseHref(window.location.href)?.urlSearchParams?.get("variant");
2047
+ if (!variantId) return getPlpOrPdpId("shopify-product-id");
2048
+ return variantId;
2049
+ }
2050
+ const pathSegment = extractor === "shopify-product-id" ? "products" : "collections";
2051
+ const tokens = getTrimmedPathName()?.split("/");
2052
+ const idIndex = tokens?.findIndex((token) => token === pathSegment);
2053
+ if (idIndex !== void 0 && idIndex >= 0 && tokens) return decodeURIComponent(tokens[idIndex + 1]);
2054
+ return null;
2055
+ }, [getTrimmedPathName]);
2056
+ const isOnPdpPage = useCallback(() => {
2057
+ return getTrimmedPathName()?.includes("/products") ?? false;
2058
+ }, [getTrimmedPathName]);
2059
+ const isOnPlpPage = useCallback(() => {
2060
+ return (getTrimmedPathName()?.includes("/collections") && !getTrimmedPathName()?.includes("/products")) ?? false;
2061
+ }, [getTrimmedPathName]);
2062
+ const value = useMemo(() => ({
2063
+ getTrimmedPathName,
2064
+ getPlpOrPdpId,
2065
+ isOnPdpPage,
2066
+ isOnPlpPage,
2067
+ isReady
2068
+ }), [
2069
+ getTrimmedPathName,
2070
+ getPlpOrPdpId,
2071
+ isOnPdpPage,
2072
+ isOnPlpPage,
2073
+ isReady
2074
+ ]);
2075
+ return /* @__PURE__ */ jsx(ShopifyUrlContext.Provider, {
2076
+ value,
2077
+ children
2078
+ });
2079
+ };
2080
+ const useShopifyUrl = () => {
2081
+ const context = useContext(ShopifyUrlContext);
2082
+ if (!context) throw new Error("useShopifyUrl must be used within a ShopifyUrlProvider");
2083
+ return context;
2084
+ };
2085
+
2086
+ //#endregion
2087
+ //#region src/hooks/useShopifyUrlOperations.ts
2088
+ const useShopifyUrlOperations = () => {
2089
+ const { getTrimmedPathName, getPlpOrPdpId, isOnPdpPage, isOnPlpPage, isReady } = useShopifyUrl();
2090
+ return {
2091
+ getTrimmedPathName,
2092
+ getPlpOrPdpId,
2093
+ isOnPdpPage,
2094
+ isOnPlpPage,
2095
+ isReady
2096
+ };
2097
+ };
2098
+ const useCurrentPageType = () => {
2099
+ const { isOnPdpPage, isOnPlpPage, isReady } = useShopifyUrl();
2100
+ return {
2101
+ getPageType: useCallback(() => {
2102
+ if (!isReady) return "unknown";
2103
+ if (isOnPdpPage()) return "pdp";
2104
+ if (isOnPlpPage()) return "plp";
2105
+ return "other";
2106
+ }, [
2107
+ isReady,
2108
+ isOnPdpPage,
2109
+ isOnPlpPage
2110
+ ]),
2111
+ isReady
2112
+ };
2113
+ };
2114
+ const useProductId = (extractor) => {
2115
+ const { getPlpOrPdpId, isReady } = useShopifyUrl();
2116
+ return {
2117
+ productId: useCallback(() => {
2118
+ if (!isReady) return null;
2119
+ return getPlpOrPdpId(extractor);
2120
+ }, [
2121
+ isReady,
2122
+ getPlpOrPdpId,
2123
+ extractor
2124
+ ]),
2125
+ isReady
2126
+ };
2127
+ };
2128
+
2129
+ //#endregion
2130
+ //#region src/hooks/useSnapCalculator.ts
2131
+ const useSnapCalculator = (snaps, maxHeight, unit) => {
2132
+ const viewportHeightPx = document.documentElement.clientHeight;
2133
+ const swipeviewHeightPx = unit === "percent" ? Math.floor(viewportHeightPx * (maxHeight / 100)) : maxHeight;
2134
+ const snapsToPixels = useMemo(() => snaps?.map((snap) => Math.abs((unit === "percent" ? Math.floor(swipeviewHeightPx * (snap / 100)) : snap) - swipeviewHeightPx)), [viewportHeightPx]);
2135
+ const getPixelToSnap = (pixels) => {
2136
+ const snapIdx = snapsToPixels?.indexOf(pixels) || 0;
2137
+ return snaps?.[snapIdx] || 0;
2138
+ };
2139
+ const getSnapToPixel = (snap) => {
2140
+ const snapIdx = snaps?.indexOf(snap) || 0;
2141
+ return snapsToPixels?.[snapIdx] || 0;
2142
+ };
2143
+ return {
2144
+ viewportHeightPx,
2145
+ snapsToPixels,
2146
+ swipeviewHeightPx,
2147
+ getPixelToSnap,
2148
+ getSnapToPixel
2149
+ };
2150
+ };
2151
+
2152
+ //#endregion
2153
+ //#region src/hooks/useSystemSettingsContext.ts
2154
+ const useSystemSettingsContext = () => {
2155
+ const context = useContext(SystemSettingsContext);
2156
+ if (!context) throw new Error("useSystemSettingsContext must be used within a SystemSettingsContextProvider");
2157
+ return { ...context };
2158
+ };
2159
+
2160
+ //#endregion
2161
+ //#region src/hooks/useTrackComponentVisibleEvent.ts
2162
+ /**
2163
+ * Tracks a component and logs an event to Amplitude when the component is visible.
2164
+ *
2165
+ * @param component - The component to track.
2166
+ * @param element - The element to track visibility of.
2167
+ * @param eventProps - Additional properties to include with the event.
2168
+ * @param eventName - The Amplitude event name to track (defaults to ChatComponentVisible).
2169
+ */
2170
+ const useTrackComponentVisibleEvent = (component, element, eventProps, eventName = SpiffyMetricsEventName.ChatComponentVisible) => {
2171
+ const isVisible = useIntersection(element, "0px");
2172
+ const hasTrackedEvent = useRef(false);
2173
+ const { trackEvent } = useAmplitude();
2174
+ const componentProps = (() => {
2175
+ if (eventName === SpiffyMetricsEventName.ChatComponentVisible) return {
2176
+ chat_component: component,
2177
+ ...eventProps
2178
+ };
2179
+ if (eventName === SpiffyMetricsEventName.SearchComponentVisible) return {
2180
+ search_component: component,
2181
+ ...eventProps
2182
+ };
2183
+ return {
2184
+ component,
2185
+ ...eventProps
2186
+ };
2187
+ })();
2188
+ useEffect(() => {
2189
+ if (isVisible && !hasTrackedEvent.current) {
2190
+ trackEvent({
2191
+ eventName,
2192
+ eventProps: componentProps
2193
+ });
2194
+ hasTrackedEvent.current = true;
2195
+ }
2196
+ }, [
2197
+ isVisible,
2198
+ component,
2199
+ eventProps,
2200
+ eventName,
2201
+ componentProps,
2202
+ trackEvent
2203
+ ]);
2204
+ };
2205
+
2206
+ //#endregion
2207
+ //#region src/hooks/useUpdateAnalyticsProps.ts
2208
+ /**
2209
+ * Updates the default analytics properties whenever the variant info changes. This hook also
2210
+ * triggers any events that should be sent once per page visit.
2211
+ */
2212
+ const useUpdateAnalyticsProps = () => {
2213
+ const variantInfo = useAtomValue(variantInfoAtom);
2214
+ const hasInitialized = useRef(false);
2215
+ const hasParsedVariantInfo = useAtomValue(hasParsedVariantInfoAtom);
2216
+ const { trackEvent, setSupplementalDefaultProps } = useAmplitude();
2217
+ useEffect(() => {
2218
+ const variantInfoWithPrefix = Object.fromEntries(Object.entries(variantInfo).map(([key, value]) => [`variantInfo.${key}`, value]));
2219
+ const defaultEventProperties = {
2220
+ page_variant: variantInfo.variant,
2221
+ ...variantInfoWithPrefix
2222
+ };
2223
+ if (variantInfo.variant === "pdp") defaultEventProperties.product_id = variantInfo.productId;
2224
+ if (variantInfo.variant === "plp") defaultEventProperties.plp_id = variantInfo.plpId;
2225
+ if (variantInfo.variant === "page_visit") {
2226
+ defaultEventProperties.page_visit_category = variantInfo.pageVisitCategory;
2227
+ defaultEventProperties.page_visit_url = variantInfo.url;
2228
+ }
2229
+ setSupplementalDefaultProps(defaultEventProperties);
2230
+ if (!hasInitialized.current && hasParsedVariantInfo) {
2231
+ trackEvent({ eventName: SpiffyMetricsEventName.BundleLoaded });
2232
+ hasInitialized.current = true;
2233
+ }
2234
+ }, [
2235
+ variantInfo,
2236
+ hasParsedVariantInfo,
2237
+ trackEvent,
2238
+ setSupplementalDefaultProps
2239
+ ]);
2240
+ };
2241
+
2242
+ //#endregion
2243
+ //#region src/contexts/chatContext.tsx
2244
+ /**
2245
+ * Record the chat assistant response in Amplitude
2246
+ *
2247
+ * @param startTimeMs The start time of the assistant response
2248
+ * @param payload The payload used to generate the response
2249
+ */
2250
+ const recordAssistantResponse = (startTimeMs, payload, track) => {
2251
+ const atomStore = getAtomStore();
2252
+ const chatState = atomStore.get(chatAtom);
2253
+ const chatSearchState = atomStore.get(chatSearchStateAtom);
2254
+ const searchProducts = atomStore.get(chatSearchProducts);
2255
+ const searchProductsSort = atomStore.get(chatSearchProductSortingAtom);
2256
+ const assistantResponseTimeMs = {
2257
+ start: startTimeMs,
2258
+ end: Date.now()
2259
+ };
2260
+ let userQueryProperty;
2261
+ if (chatState.replyEventCategory === UserEventCategory.SuggestionClicked && chatState.suggestion) userQueryProperty = chatState.suggestion.content;
2262
+ else if (chatState.userQuery && chatState.userQuery.length > 0) userQueryProperty = chatState.userQuery;
2263
+ const eventProps = {
2264
+ response_time_ms: assistantResponseTimeMs.end - assistantResponseTimeMs.start,
2265
+ user_event_type: chatState.replyEventCategory,
2266
+ user_query: userQueryProperty
2267
+ };
2268
+ if (chatState.replyEventCategory === UserEventCategory.FormSubmitted) {
2269
+ const lastAssistantTurn = chatState.messages.filter((turn) => turn.length > 0 && turn[0].role === MessageRole.Assistant).pop();
2270
+ const formType = payload.userEvents?.find((event) => event.category === UserEventCategory.FormSubmitted)?.attributes.formType;
2271
+ const formStatus = lastAssistantTurn?.some((response) => response.type === MessageType.Order);
2272
+ eventProps.form_submitted_attributes = {
2273
+ form_type: formType,
2274
+ status: formStatus ? "success" : "failed"
2275
+ };
2276
+ }
2277
+ if (chatSearchState === "product-page") {
2278
+ eventProps.search_products_returned = searchProducts.length;
2279
+ eventProps.search_products_sort_type = searchProductsSort;
2280
+ }
2281
+ track(SpiffyMetricsEventName.ChatAssistantResponse, { eventProps });
2282
+ };
2283
+ const ChatContext = createContext(void 0);
2284
+ const updateMessageState = (message, lastMessage, setMessages) => {
2285
+ if (lastMessage == null) {
2286
+ setMessages((prev) => [...prev, [message]]);
2287
+ return message;
2288
+ }
2289
+ if (lastMessage.type === MessageType.Text && message.type === MessageType.Text) {
2290
+ const newMessage = {
2291
+ ...lastMessage,
2292
+ metadata: {
2293
+ ...lastMessage.metadata,
2294
+ content: lastMessage.metadata.content + message.metadata.content
2295
+ }
2296
+ };
2297
+ setMessages((prev) => {
2298
+ const lastTurn = prev[prev.length - 1];
2299
+ return [...prev.slice(0, prev.length - 1), [...lastTurn.slice(0, lastTurn.length - 1), newMessage]];
2300
+ });
2301
+ return newMessage;
2302
+ }
2303
+ setMessages((prev) => [...prev.slice(0, prev.length - 1), [...prev[prev.length - 1], message]]);
2304
+ return message;
2305
+ };
2306
+ const handleStreamingError = (_error, setRequestFailure, setMessages) => {
2307
+ setRequestFailure(true);
2308
+ setMessages((prev) => [...prev, [{
2309
+ id: v4(),
2310
+ role: MessageRole.Assistant,
2311
+ type: MessageType.Text,
2312
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
2313
+ metadata: { content: "I'm sorry! I'm having trouble right now. Please refresh the page or try again in a moment." }
2314
+ }]]);
2315
+ };
2316
+ const processStreamingResponse = async (stream, messageInterceptor, handleSearchResults, setMessages, setSearchIsLoading, chatId) => {
2317
+ let lastMessage;
2318
+ let hasSearchResults = false;
2319
+ for await (const response of stream) try {
2320
+ if (messageInterceptor.intercept(response)) return { hasSearchResults };
2321
+ const message = messageFromResponse(response);
2322
+ if (!message) throw new Error("Failed to transform API response to client message");
2323
+ if (message.type === MessageType.ProductSearch) {
2324
+ handleSearchResults(message);
2325
+ hasSearchResults = true;
2326
+ setSearchIsLoading(false);
2327
+ }
2328
+ lastMessage = updateMessageState(message, lastMessage, setMessages);
2329
+ } catch (error) {
2330
+ logger_default.logWarn(`[spiffy-ai] Failed to generate responses from stream chat_id=${chatId}`, error, {
2331
+ lastResponse: lastMessage,
2332
+ response
2333
+ });
2334
+ }
2335
+ return { hasSearchResults };
2336
+ };
2337
+ const ChatContextProvider = ({ children }) => {
2338
+ const logPerfMetric = useSetAtom(logPerfMetricAtom);
2339
+ const [widgetInitialized, setWidgetInitialized] = useState(false);
2340
+ const setUserHasReplied = useSetAtom(userHasRepliedAtom);
2341
+ const [messages, setMessages] = useAtom(messagesAtom);
2342
+ const setUserEvents = useSetAtom(userEventsAtom);
2343
+ const setSuggestions = useSetAtom(suggestionsAtom);
2344
+ const [suggestionsLoading, setSuggestionsLoading] = useAtom(suggestionsLoadingAtom);
2345
+ const [responseStreaming, setResponseStreaming] = useAtom(responseStreamingAtom);
2346
+ const setRequestFailure = useSetAtom(requestFailureAtom);
2347
+ const userEvents = useAtomValue(userEventQueueAtom);
2348
+ const userQueueEventCount = useAtomValue(userQueueEventCountAtom);
2349
+ const markUserEventsProcessed = useSetAtom(processUserEventAtom);
2350
+ const clearUserEventQueue = useSetAtom(clearUserEventAtom);
2351
+ const userId = useAtomValue(userIdAtom);
2352
+ const chatId = useAtomValue(chatIdAtom);
2353
+ const supportedEvent = useAtomValue(supportedEventAtom);
2354
+ const orgId = "mock-org-id";
2355
+ const variantInfo = useAtomValue(variantInfoAtom);
2356
+ const settingsContext = useSystemSettingsContext();
2357
+ const messageInterceptor = useMessageInterceptor();
2358
+ const handleSearchResults = useSetAtom(handleSearchResultsAtom);
2359
+ const setSearchIsLoading = useSetAtom(chatSearchIsLoadingAtom);
2360
+ const { track } = useAmplitudeTracking();
2361
+ const getStreamingResponses = useCallback(async (payload) => {
2362
+ logPerfMetric(PerfMetricsEvents.FirstResponseStarted);
2363
+ const stream = commerce_api_default.getNextResponseStreaming(payload);
2364
+ try {
2365
+ setRequestFailure(false);
2366
+ const { hasSearchResults } = await processStreamingResponse(stream, messageInterceptor, handleSearchResults, setMessages, setSearchIsLoading, chatId);
2367
+ return { hasSearchResults };
2368
+ } catch (e) {
2369
+ handleStreamingError(e, setRequestFailure, setMessages);
2370
+ throw e;
2371
+ } finally {
2372
+ logPerfMetric(PerfMetricsEvents.FirstResponseCompleted);
2373
+ }
2374
+ }, [
2375
+ logPerfMetric,
2376
+ setRequestFailure,
2377
+ messageInterceptor,
2378
+ handleSearchResults,
2379
+ setMessages,
2380
+ setSearchIsLoading,
2381
+ chatId
2382
+ ]);
2383
+ const getSuggestions = useCallback(async () => {
2384
+ logPerfMetric(PerfMetricsEvents.FirstSuggestionsStarted);
2385
+ setSuggestionsLoading(true);
2386
+ setSuggestions([]);
2387
+ const payloadWithoutAppLoaded = createResponsePayload({
2388
+ userEvents: [],
2389
+ generationParams: settingsContext.generationParams
2390
+ });
2391
+ const response = await commerce_api_default.getNextSuggestions(payloadWithoutAppLoaded);
2392
+ setSuggestions(response.sort((a, b) => a.content.length - b.content.length));
2393
+ setSuggestionsLoading(false);
2394
+ logPerfMetric(PerfMetricsEvents.FirstSuggestionsCompleted);
2395
+ }, [
2396
+ logPerfMetric,
2397
+ setSuggestionsLoading,
2398
+ setSuggestions,
2399
+ settingsContext.generationParams
2400
+ ]);
2401
+ const getResponses = useCallback(async (payload) => {
2402
+ try {
2403
+ const requestPayload = payload ?? createResponsePayload({
2404
+ userEvents,
2405
+ generationParams: settingsContext.generationParams
2406
+ });
2407
+ setResponseStreaming(true);
2408
+ setSuggestions([]);
2409
+ const startTimeMs = Date.now();
2410
+ await getStreamingResponses(requestPayload);
2411
+ recordAssistantResponse(startTimeMs, requestPayload, track);
2412
+ await getSuggestions();
2413
+ } catch (error) {
2414
+ logger_default.logError("[spiffy-ai] getResponses error", error);
2415
+ } finally {
2416
+ markUserEventsProcessed(userEvents.map(({ eventId }) => eventId));
2417
+ setUserHasReplied(false);
2418
+ setResponseStreaming(false);
2419
+ }
2420
+ }, [
2421
+ userEvents,
2422
+ settingsContext.generationParams,
2423
+ setResponseStreaming,
2424
+ setSuggestions,
2425
+ getStreamingResponses,
2426
+ markUserEventsProcessed,
2427
+ getSuggestions,
2428
+ setUserHasReplied,
2429
+ track
2430
+ ]);
2431
+ useEffect(() => {
2432
+ const processUserEvents = async () => {
2433
+ if (responseStreaming || !widgetInitialized) return;
2434
+ if (variantInfo.variant === "pdp" && !variantInfo.productId || variantInfo.variant === "plp" && !variantInfo.plpId || variantInfo.variant === "page_visit" && !variantInfo.url) {
2435
+ logger_default.logDebug("[spiffy-ai] variantInfo has invalid values, skipping...", {
2436
+ variantInfo,
2437
+ supportedEvent
2438
+ });
2439
+ return;
2440
+ }
2441
+ logger_default.logDebug(`Assistants Turn is_currently_streaming=${responseStreaming} initialized=${widgetInitialized}`);
2442
+ try {
2443
+ await getResponses();
2444
+ logger_default.logInfo(`Assistants Turn [finished]`);
2445
+ } catch (error) {
2446
+ logger_default.logError("[spiffy-ai] Assistants Turn error", error);
2447
+ }
2448
+ };
2449
+ if (userQueueEventCount > 0) processUserEvents();
2450
+ }, [
2451
+ getResponses,
2452
+ responseStreaming,
2453
+ userQueueEventCount,
2454
+ widgetInitialized,
2455
+ variantInfo,
2456
+ supportedEvent
2457
+ ]);
2458
+ useEffect(() => {
2459
+ if (widgetInitialized || responseStreaming) {
2460
+ logger_default.logDebug(`[spiffy-ai] initializeWidget [skipped] is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`);
2461
+ return;
2462
+ }
2463
+ const hydrateChat = async () => {
2464
+ try {
2465
+ logger_default.logDebug(`[spiffy-ai] initializeWidget is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`);
2466
+ const { messages: existingMessages, userEvents: userEvents$1 } = await commerce_api_default.getResponses(orgId, chatId, userId);
2467
+ setMessages([...existingMessages]);
2468
+ setUserEvents([...userEvents$1]);
2469
+ getResponses();
2470
+ } catch (error) {
2471
+ logger_default.logInfo(`Init chat [exception] chat_id=${chatId} error=${error}`, error);
2472
+ if (error instanceof SessionRestartRequired) {
2473
+ const appLoadedEvent = createAppLoadedEvent();
2474
+ const visitEvent = createVisitUserEvent({ variantInfo });
2475
+ setMessages([]);
2476
+ clearUserEventQueue();
2477
+ if (visitEvent) {
2478
+ const payload = createResponsePayload({
2479
+ userEvents: [appLoadedEvent, visitEvent],
2480
+ generationParams: settingsContext.generationParams
2481
+ });
2482
+ getResponses(payload);
2483
+ }
2484
+ }
2485
+ } finally {
2486
+ setWidgetInitialized(true);
2487
+ }
2488
+ };
2489
+ hydrateChat();
2490
+ }, []);
2491
+ const onFocus = useCallback(async () => {
2492
+ try {
2493
+ if (!responseStreaming && !suggestionsLoading && orgId) {
2494
+ const { messages: existingMessages } = await commerce_api_default.getResponses(orgId, chatId, userId);
2495
+ if (existingMessages.length > messages.length) setMessages([...existingMessages]);
2496
+ }
2497
+ } catch (error) {
2498
+ logger_default.logError("[spiffy-ai] onFocus error", error);
2499
+ }
2500
+ }, [
2501
+ responseStreaming,
2502
+ suggestionsLoading,
2503
+ orgId,
2504
+ chatId,
2505
+ userId,
2506
+ messages.length,
2507
+ setMessages
2508
+ ]);
2509
+ useEffect(() => {
2510
+ window.addEventListener("focus", onFocus);
2511
+ return () => {
2512
+ window.removeEventListener("focus", onFocus);
2513
+ };
2514
+ }, [onFocus]);
2515
+ const chatContext = useMemo(() => ({}), []);
2516
+ return /* @__PURE__ */ jsx(ChatContext.Provider, {
2517
+ value: chatContext,
2518
+ children
2519
+ });
2520
+ };
2521
+
2522
+ //#endregion
2523
+ //#region src/contexts/enviveCssContext.tsx
2524
+ const EnviveCssProvider = ({ children }) => {
2525
+ const { colorsConfig, frontendConfig, loading } = useNewOrgConfig();
2526
+ let merchantThemeCss = `* {}`;
2527
+ if (colorsConfig && !loading) merchantThemeCss = `
2528
+ * {
2529
+ --spiffy-colors-text-primary: ${colorsConfig.textPrimary};
2530
+ --spiffy-colors-text-secondary: ${colorsConfig.textSecondary};
2531
+ --spiffy-colors-text-accent: ${colorsConfig.textAccent};
2532
+ --spiffy-colors-text-link: ${colorsConfig.textLink};
2533
+ --spiffy-colors-text-light: ${colorsConfig.textLight};
2534
+ --spiffy-colors-background-primary: ${colorsConfig.backgroundPrimary};
2535
+ --spiffy-colors-background-secondary: ${colorsConfig.backgroundSecondary};
2536
+ --spiffy-colors-background-secondary-dark: ${colorsConfig.backgroundSecondaryDark};
2537
+ --spiffy-colors-background-tertiary: ${colorsConfig.backgroundTertiary};
2538
+ --spiffy-colors-background-dark: ${colorsConfig.backgroundDark};
2539
+ --spiffy-colors-background-light: ${colorsConfig.backgroundLight};
2540
+ --spiffy-colors-background-saturated: ${colorsConfig.backgroundSaturated};
2541
+ --spiffy-colors-border-light: ${colorsConfig.borderLight};
2542
+ --spiffy-colors-border-medium: ${colorsConfig.borderMedium};
2543
+ --spiffy-colors-border-dark: ${colorsConfig.borderDark};
2544
+ --spiffy-colors-border-outline: ${colorsConfig.borderOutline};
2545
+ --spiffy-colors-accent-primary: ${colorsConfig.accentPrimary};
2546
+ --spiffy-colors-accent-secondary: ${colorsConfig.accentSecondary};
2547
+ }`;
2548
+ console.log(frontendConfig);
2549
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
2550
+ frontendConfig ? /* @__PURE__ */ jsx("style", {
2551
+ id: "merchant-css-overrides",
2552
+ children: `${frontendConfig.merchantOverrideCss}`
2553
+ }) : null,
2554
+ colorsConfig ? /* @__PURE__ */ jsx("style", {
2555
+ id: "merchant-css-colors",
2556
+ children: `${merchantThemeCss}`
2557
+ }) : null,
2558
+ children
2559
+ ] });
2560
+ };
2561
+
2562
+ //#endregion
2563
+ //#region src/contexts/systemSettingsContext.tsx
2564
+ const getChatModelName = () => {
2565
+ const urlObj = new URL(window.location.href);
2566
+ const params = new URLSearchParams(urlObj.search);
2567
+ return Object.fromEntries(params.entries()).llm_model_name;
2568
+ };
2569
+ const defaultGenerationParams = {
2570
+ stream: true,
2571
+ numSuggestions: 3,
2572
+ model: getChatModelName()
2573
+ };
2574
+ const SystemSettingsContext = createContext(void 0);
2575
+ const SystemSettingsContextProvider = ({ children, generationParams, showDebugBar }) => {
2576
+ const [params, setParams] = useState(() => generationParams ?? defaultGenerationParams);
2577
+ const endpointURL = useAtomValue(baseUrlAtom);
2578
+ const settingsContextValues = useMemo(() => ({
2579
+ generationParams: params,
2580
+ showDebugBar,
2581
+ setGenerationParams: setParams,
2582
+ endpointURL
2583
+ }), [
2584
+ generationParams,
2585
+ endpointURL,
2586
+ showDebugBar
2587
+ ]);
2588
+ return /* @__PURE__ */ jsx(SystemSettingsContext.Provider, {
2589
+ value: settingsContextValues,
2590
+ children
2591
+ });
2592
+ };
2593
+
2594
+ //#endregion
2595
+ export { CdnProvider, ChatContext, ChatContextProvider, EnviveCssProvider, FeatureFlagServiceProvider, GraphQLProvider, NewOrgConfigProvider, SearchProvider, SearchResultsState, SessionStorageProvider, ShopifyUrlProvider, SystemSettingsContext, SystemSettingsContextProvider, UserIdentityProvider, createAppLoadedEvent, createVisitUserEvent, defaultGenerationParams, getSearchResultsState, isElementPartiallyVisible, isWithinBusinessHours, useAmplitudeTracking, useAssetUrl, useBlockBackButton, useCdn, useCdnBasePath, useCdnUrl, useChatToggle, useChatToggleAnalytics, useColorsAndFrontendConfig, useCurrentPageType, useCustomerSupportHandoff, useDebounce, useElementObserver, useEnviveFeatureFlag, useFeatureFlagService, useGrabAndScroll, useGraphQLClient, useIdentifyUser, useImageResolver, useIntersection, useIsSmallScreen, useLocalStorageListener, useLocalStorageValue, useMessageFilter, useMessageScrollObserver, useNewOrgConfig, useNewOrgConfigContext, useOrgId, useProductId, useProductSearch, useSearch, useSearchService, useSearchWithQuery, useSessionStorage, useSessionStorageValue, useShopifyUrl, useShopifyUrlOperations, useSnapCalculator, useSpiffyFeatureFlag, useSystemSettingsContext, useTrackComponentVisibleEvent, useUpdateAnalyticsProps, useUserIdentity };
2596
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"contexts-DGlr7M1o.js","names":["CdnProvider: React.FC<{ children: React.ReactNode }>","errorResponseBody","throwSessionRestartRequiredIf","config: Configuration","context: Context","err: unknown","error: unknown","data: V1GetSessionMessages200Response","responses: Response[][]","suggestions: Suggestion[]","userEvents: UserEvent[]","assistantMessages: Message[][]","userMessages: Message[][]","request: V1OrgConfigGetRequest","UserIdentityProvider: React.FC<{ children: React.ReactNode }>","uuid","CommerceApiClient","listener: LocalStorageEventListener","FeatureFlagServiceProvider: React.FC<\n  FeatureFlagServiceProviderProps\n>","NewOrgConfigProvider: React.FC<NewOrgConfigProviderProps>","uuid","handleSelectFilterItem: SelectFilterItem","enviveOrgShortNameAtom","enviveEnvAtom","SearchProvider: React.FC<{ children: React.ReactNode }>","config: Configuration","SessionStorageProvider: React.FC<{\n  children: React.ReactNode;\n}>","value","ShopifyUrlProvider: React.FC<{ children: React.ReactNode }>","defaultEventProperties: Record<string, unknown>","userQueryProperty: string | undefined","eventProps: Record<string, unknown>","uuid","lastMessage: Message | undefined","error: unknown","CommerceApiClient","userEvents","EnviveCssProvider: React.FC<EnviveCssProviderProps>","defaultGenerationParams: GenerationParams"],"sources":["../src/contexts/cdnContext.tsx","../src/types/exceptions/sessionExceptions.ts","../src/types/exceptions/unsupportedProductExceptions.ts","../src/application/commerce-api.ts","../src/hooks/useAmplitudeOperations.ts","../src/hooks/useBlockBackButton.ts","../src/hooks/useCdnOperations.ts","../src/hooks/useChatToggle.ts","../src/hooks/useChatToggleAnalytics.ts","../src/hooks/useCustomerSupportHandoff.ts","../src/hooks/useDebounce.ts","../src/hooks/useElementObserver.ts","../src/hooks/useGrabAndScroll.ts","../src/contexts/graphqlContext.tsx","../src/hooks/useGraphQLConfig.ts","../src/contexts/userIdentityContext.tsx","../src/hooks/useIdentifyUser.ts","../src/hooks/useImageResolver.ts","../src/hooks/useIntersection.ts","../src/hooks/useIsSmallScreen.ts","../src/hooks/useLocalStorageOperations.ts","../src/hooks/useMessageFilter.ts","../src/hooks/useMessageScrollObserver.ts","../src/contexts/featureFlagServiceContext.tsx","../src/contexts/newOrgConfigContext.tsx","../src/hooks/useNewOrgConfig.ts","../src/hooks/utils.ts","../src/hooks/useSearch.tsx","../src/hooks/useAppDetails.ts","../src/contexts/searchContext.tsx","../src/hooks/useSearchOperations.ts","../src/contexts/sessionStorageContext.tsx","../src/hooks/useSessionStorageOperations.ts","../src/contexts/shopifyUrlContext.tsx","../src/hooks/useShopifyUrlOperations.ts","../src/hooks/useSnapCalculator.ts","../src/hooks/useSystemSettingsContext.ts","../src/hooks/useTrackComponentVisibleEvent.ts","../src/hooks/useUpdateAnalyticsProps.ts","../src/contexts/chatContext.tsx","../src/contexts/enviveCssContext.tsx","../src/contexts/systemSettingsContext.tsx"],"sourcesContent":["import React, { createContext, useContext, useCallback, useMemo } from \"react\";\nimport { useAtomValue } from \"jotai\";\nimport { cdnUrlAtom } from \"../atoms/envive/enviveConfig\";\n\ninterface CdnContextType {\n  cdnUrl: string;\n  getCdnBasePath: () => string;\n  getAssetURL: (assetName: string, orgShortName: string) => string;\n}\n\nconst CdnContext = createContext<CdnContextType | null>(null);\n\nexport const CdnProvider: React.FC<{ children: React.ReactNode }> = ({\n  children,\n}) => {\n  const cdnUrlAtomValue = useAtomValue(cdnUrlAtom);\n  const cdnUrl = cdnUrlAtomValue || \"https://cdn.spiffy.ai/other\";\n\n  const getCdnBasePath = useCallback(() => {\n    return cdnUrl;\n  }, [cdnUrl]);\n\n  const getAssetURL = useCallback(\n    (assetName: string, orgShortName: string) => {\n      return `${getCdnBasePath()}/assets/${orgShortName}/${assetName}`;\n    },\n    [getCdnBasePath]\n  );\n\n  const value = useMemo(\n    () => ({\n      cdnUrl,\n      getCdnBasePath,\n      getAssetURL,\n    }),\n    [cdnUrl, getCdnBasePath, getAssetURL]\n  );\n\n  return <CdnContext.Provider value={value}>{children}</CdnContext.Provider>;\n};\n\nexport const useCdn = () => {\n  const context = useContext(CdnContext);\n  if (!context) {\n    throw new Error(\"useCdn must be used within a CdnProvider\");\n  }\n  return context;\n};\n","export class SessionRestartRequired extends Error {\n  constructor() {\n    super('Session restart required');\n    this.name = 'SessionRestartRequired';\n  }\n}\n","export class UnsupportedProductException extends Error {\n  constructor() {\n    super('Unsupported product');\n    this.name = 'UnsupportedProduct';\n  }\n}","import Logger from \"src/application/logging/logger\";\nimport {\n  Configuration,\n  CustomerServiceApi,\n  DefaultApi,\n  InferenceApi,\n  ReportSessionRequest,\n  ResponseError,\n  UserEventCategory,\n  V1GetSessionMessages200Response,\n  V1OrgConfigGetRequest,\n  ContextSourceEnum,\n  V1OrgConfigGetSourceEnum,\n  CustomerServiceProvider,\n  ResponseCategory,\n  FormType,\n} from \"@spiffy-ai/commerce-api-client\";\nimport { validateSuggestion } from \"src/application/models/validators/validateSuggestion\";\nimport { validateUserEvent } from \"src/application/models/validators/validateUserEvent\";\nimport {\n  messageFromQueryEvent,\n  messageFromResponse,\n  messageFromSuggestionEvent,\n} from \"src/application/utils\";\nimport {\n  Message,\n  NextMessageRequest,\n  Response,\n  Suggestion,\n  SupportedEventRequest,\n  UserEvent,\n} from \"src/application/models\";\nimport { messageRequestToCommerceMessageRequest } from \"src/application/utils/nextMessageRequestToApiRequest\";\nimport { validateResponse } from \"src/application/models/validators/validateResponse\";\nimport { coreSupportedEventRequestToApiRequest } from \"src/application/utils/supportedEventRequestToApiRequest\";\nimport { SessionRestartRequired } from \"src/types/exceptions/sessionExceptions\";\nimport { UnsupportedProductException } from \"src/types/exceptions/unsupportedProductExceptions\";\nimport { ClientDetails } from \"src/application/models/clientDetails\";\nimport { getAtomStore } from \"src/atoms/atomStore/atomStore\";\nimport { ProductExperiment } from \"src/application/models/productExperiment\";\nimport { FeatureGates } from \"src/application/models/featureGates\";\nimport { validateOrgConfigResults } from \"src/application/models/validators/validateOrgConfigResults\";\nimport { OrgConfig } from \"src/application/models/api/orgConfigResults\";\nimport { SupportedEvent } from \"src/atoms/app/variant\";\nimport { messageFromFormSubmittedEvent } from \"src/application/utils/messageFromFormSubmittedEvent\";\nimport type { Context } from \"@spiffy-ai/commerce-api-client/dist/models/Context\";\nimport { ContextEnvEnum } from \"@spiffy-ai/commerce-api-client\"; // Import ContextEnvEnum\nimport {\n  baseUrlAtom,\n  reactAppNameAtom,\n  contextSourceAtom,\n  orgShortNameAtom,\n  envAtom, // Import envAtom\n} from \"src/atoms/envive/enviveConfig\";\nimport { userIdAtom, chatIdAtom } from \"src/atoms/app\"; // Import userIdAtom and chatIdAtom\nimport { orgIdAtom, featureFlagServiceAtom } from \"src/atoms/org/graphqlConfig\"; // Import new orgIdAtom and featureFlagServiceAtom\n\nasync function errorResponseBody(error: ResponseError) {\n  try {\n    return await error.response.json();\n  } catch {\n    return {};\n  }\n}\n\nasync function throwSessionRestartRequiredIf(errorMsg: string, error: unknown) {\n  if (!(error instanceof ResponseError)) {\n    Logger.logInfo(errorMsg, error);\n    throw error;\n  }\n\n  const errorResponse = await errorResponseBody(error);\n  if (\n    errorResponse?.message?.toLowerCase() === \"unsupported product\" || // for backward compatibility. newer versions of the API return sub_code instead of message\n    errorResponse?.app_code?.toUpperCase() === \"PRODUCT_NOT_FOUND\"\n  ) {\n    throw new UnsupportedProductException();\n  } else if (\n    errorResponse?.app_code?.toUpperCase() === \"RESTART_SESSION\" ||\n    errorResponse?.sub_code?.toUpperCase() === \"NOT_FOUND\" // for backward compatibility. new API responses will contain \"app_code\"\n  ) {\n    Logger.logInfo(\n      \"Session does not exist. Re-start session\",\n      error,\n      error.response,\n      errorResponse\n    );\n    throw new SessionRestartRequired();\n  }\n\n  Logger.logInfo(errorMsg, error);\n  throw error;\n}\n\nclass CommerceApiClient {\n  private readonly defaultApi: DefaultApi;\n\n  private readonly customerServiceApi: CustomerServiceApi;\n\n  private readonly inferenceApi: InferenceApi;\n\n  private static instance: CommerceApiClient | undefined;\n\n  private suggestionsAbortController = new AbortController();\n\n  private responsesAbortController = new AbortController();\n\n  private static getInstance = (): CommerceApiClient => {\n    if (!CommerceApiClient.instance) {\n      CommerceApiClient.instance = new CommerceApiClient();\n    }\n\n    return CommerceApiClient.instance;\n  };\n\n  private constructor(basePath?: string) {\n    const atomStore = getAtomStore();\n    const baseUrl = atomStore.get(baseUrlAtom);\n    const path = basePath || baseUrl;\n    // API Key is now handled at the EnviveConfigProvider level\n    const config: Configuration = new Configuration({\n      basePath: path,\n      headers: {\n        \"Content-Type\": \"application/json\",\n        Accept: \"application/json\",\n      },\n    });\n    this.defaultApi = new DefaultApi(config);\n    this.inferenceApi = new InferenceApi(config);\n    this.customerServiceApi = new CustomerServiceApi(config);\n  }\n\n  static resolveUrl = async (url: string) => {\n    const atomStore = getAtomStore();\n    const orgShortName = atomStore.get(orgShortNameAtom);\n    const orgId = atomStore.get(orgIdAtom);\n    const userId = atomStore.get(userIdAtom);\n    const chatId = atomStore.get(chatIdAtom);\n    const source = atomStore.get(contextSourceAtom);\n    const env = atomStore.get(envAtom);\n\n    const featureFlagService = atomStore.get(featureFlagServiceAtom);\n\n    const context: Context = {\n      user_id: userId ?? \"\",\n      org_id: orgId ?? \"\",\n      org_short_name: orgShortName ?? \"\",\n      chat_id: chatId ?? \"\",\n      source: source ?? ContextSourceEnum.App,\n      env: (env as ContextEnvEnum) ?? ContextEnvEnum.Dev, // Cast env to ContextEnvEnum\n    };\n\n    const featureGates =\n      featureFlagService?.featureFlagService?.getFeatureFlags() || {};\n    const urlResolvingRequest = {\n      url,\n      context,\n      feature_gates: featureGates,\n    };\n\n    const rawResponse =\n      await CommerceApiClient.getInstance().inferenceApi.v1UrlResolvingPostRaw({\n        UrlResolvingRequest: urlResolvingRequest,\n      });\n\n    const responseBody = await rawResponse.raw.json();\n\n    return responseBody;\n  };\n\n  static reportSession = async (\n    reportRequest: ReportSessionRequest\n  ): Promise<void> => {\n    await CommerceApiClient.getInstance().defaultApi.v1ChatsReportSessionIdPost(\n      {\n        ReportSessionRequest: reportRequest,\n      }\n    );\n  };\n\n  static getNextResponses = async (\n    payload: NextMessageRequest\n  ): Promise<Message[]> => {\n    try {\n      const response =\n        await CommerceApiClient.getInstance().inferenceApi.v1NextResponsesPost({\n          NextMessageRequest: messageRequestToCommerceMessageRequest(payload),\n        });\n      const messages = response\n        .map((resp) => validateResponse(resp))\n        .map((resp) => messageFromResponse(resp));\n\n      return messages.filter((m): m is Message => m != null);\n    } catch (err: unknown) {\n      Logger.logInfo(\"Failed to get next responses\", err, {\n        payloadContext: payload?.context,\n        userEvents: payload?.userEvents,\n      });\n      await throwSessionRestartRequiredIf(\"Failed to get next responses\", err);\n      return [];\n    }\n  };\n\n  static getNextResponseStreaming = (\n    payload: NextMessageRequest\n  ): AsyncGenerator<Response, void, unknown> => {\n    async function* generate(\n      inferenceApi: InferenceApi,\n      abortController: AbortController\n    ) {\n      // make sure streaming is enabled\n      try {\n        const response = await inferenceApi.v1NextResponsesPostRaw(\n          {\n            NextMessageRequest: messageRequestToCommerceMessageRequest(payload),\n          },\n          { signal: abortController.signal }\n        );\n\n        // Read the response as a stream of data\n        if (!response.raw.body) {\n          Logger.logError(\n            \"[spiffy-ai] No body in the streamed response\",\n            undefined,\n            {\n              response: response.raw,\n            }\n          );\n          return;\n        }\n\n        const reader = response.raw.body.getReader();\n        const decoder = new TextDecoder(\"utf-8\");\n\n        let partial = \"\";\n        // TODO this function is recreated every time new data comes from the stream - define it outside of the generator\n        const safeParse = (line: string): unknown => {\n          try {\n            return JSON.parse(line); // Parse the JSON string\n          } catch (err) {\n            Logger.logError(\"[spiffy-ai] Error parsing streamed line\", err, {\n              line,\n              partial,\n            });\n            // swallow the error and set the partial to our current chunk\n            partial = line;\n            return partial;\n          }\n        };\n\n        // TODO this function is recreated every time new data comes from the stream - define it outside of the generator\n        const processChunk = (chunk: string): unknown[] => {\n          // merge the partial with the incoming chunk\n          const lines = `${partial}${chunk}`.split(\"\\n\");\n\n          const parsedLines = lines\n            .map((line) => line.replace(/^data: /, \"\").trim()) // Remove the \"data: \" prefix\n            .filter((line) => line !== \"\" && line !== \"[DONE]\") // Remove empty lines and \"[DONE]\"\n            .map(safeParse)\n            .filter((v) => v); // and filter out the undefined values\n          return parsedLines;\n        };\n\n        while (true) {\n          // eslint-disable-next-line no-await-in-loop\n          const { done, value } = await reader.read();\n\n          if (done) {\n            break;\n          }\n\n          // Massage and parse the chunk of data\n          const chunk = decoder.decode(value);\n          const parsedLines = processChunk(chunk);\n\n          for (const parsedLine of parsedLines) {\n            const validatedResponse = validateResponse(parsedLine);\n\n            if (validatedResponse) {\n              yield validatedResponse;\n            }\n          }\n        }\n      } catch (error: unknown) {\n        Logger.logError(\n          \"[spiffy-ai] Failed to get next streaming responses\",\n          error,\n          {\n            payloadContext: payload?.context,\n            userEvents: payload?.userEvents,\n          }\n        );\n        await throwSessionRestartRequiredIf(\n          \"Failed to get next streaming responses\",\n          error\n        );\n      }\n    }\n\n    CommerceApiClient.getInstance().responsesAbortController.abort();\n    CommerceApiClient.getInstance().responsesAbortController =\n      new AbortController();\n\n    return generate(\n      CommerceApiClient.getInstance().inferenceApi,\n      CommerceApiClient.getInstance().responsesAbortController\n    );\n  };\n\n  static getNextSuggestions = async (\n    payload: NextMessageRequest\n  ): Promise<Suggestion[]> => {\n    try {\n      CommerceApiClient.getInstance().suggestionsAbortController.abort();\n      CommerceApiClient.getInstance().suggestionsAbortController =\n        new AbortController();\n\n      const response =\n        await CommerceApiClient.getInstance().inferenceApi.v1NextSuggestionsPost(\n          {\n            NextMessageRequest: messageRequestToCommerceMessageRequest(payload),\n          },\n          {\n            signal:\n              CommerceApiClient.getInstance().suggestionsAbortController.signal,\n          }\n        );\n\n      const suggestions = response\n        .map((resp) => validateSuggestion(resp))\n        .filter((suggestion): suggestion is Suggestion => suggestion != null);\n\n      return suggestions;\n    } catch (error: unknown) {\n      Logger.logInfo(\"Failed to get suggestions\", error, {\n        payloadContext: payload?.context,\n        userEvents: payload?.userEvents,\n      });\n\n      await throwSessionRestartRequiredIf(\"Failed to get suggestions\", error);\n      return [];\n    }\n  };\n\n  /**\n   * Fetches the existing chat data for a given session and transforms them to reconstruct the chat history.\n   *\n   * @param orgId The organization Id\n   * @param chatId The existing chat Id\n   * @param userId The user Id\n   *\n   * @returns A list of messages that were exchanged in the chat, throws an error if the chat session has expired.\n   */\n  static getResponses = async (\n    orgId: string,\n    chatId: string,\n    userId: string\n  ): Promise<{\n    responses: Response[][];\n    userEvents: UserEvent[];\n    suggestions: Suggestion[];\n    messages: Message[][];\n  }> => {\n    let data: V1GetSessionMessages200Response = {\n      responses: [],\n      suggestions: [],\n      user_events: [],\n    };\n    const request = {\n      org_id: orgId,\n      chat_id: chatId,\n      user_id: userId,\n    };\n    try {\n      data =\n        await CommerceApiClient.getInstance().defaultApi.v1GetSessionMessages(\n          request\n        );\n    } catch (error: unknown) {\n      await throwSessionRestartRequiredIf(\n        \"Failed to get chat responses\",\n        error\n      );\n    }\n\n    const responses: Response[][] = data?.responses?.map((turn) =>\n      turn\n        .map((response) => validateResponse(response))\n        .filter((response): response is Response => response != null)\n    );\n\n    const suggestions: Suggestion[] = data?.suggestions\n      .map((suggestion) => validateSuggestion(suggestion))\n      .filter((suggestion): suggestion is Suggestion => suggestion != null);\n\n    const userEvents: UserEvent[] = data?.user_events\n      .map((event) => validateUserEvent(event))\n      .filter((event): event is UserEvent => event != null);\n\n    // if a form has already been submitted, don't show it in the chat history\n    const formSubmittedUserEventsFormIds = userEvents\n      .filter((event) => event.category === UserEventCategory.FormSubmitted)\n      .map((event) => event.attributes.formResponseId);\n\n    const assistantMessages: Message[][] = responses\n      .map((turn) =>\n        turn\n          .filter(\n            (response) =>\n              !(\n                response.category === ResponseCategory.Form &&\n                formSubmittedUserEventsFormIds.includes(response.id)\n              )\n          )\n          .map((response) => messageFromResponse(response))\n          .filter((message): message is Message => message != null)\n      )\n      .filter((turn) => turn.length > 0);\n\n    const userMessages: Message[][] = userEvents\n      .map((event) => {\n        if (\n          [UserEventCategory.QueryTyped, UserEventCategory.Search].includes(\n            event.category\n          )\n        ) {\n          return [messageFromQueryEvent(event)];\n        }\n\n        if (event.category === UserEventCategory.SuggestionClicked) {\n          return [messageFromSuggestionEvent(event, suggestions)];\n        }\n\n        if (event.category === UserEventCategory.FormSubmitted) {\n          const formResponse = responses\n            .flat()\n            .find(\n              (response) =>\n                response.id === event.attributes.formResponseId &&\n                event.attributes.formType !== FormType.Escalation\n            );\n\n          if (formResponse && formResponse.category === ResponseCategory.Form) {\n            return [\n              messageFromFormSubmittedEvent(event, formResponse.attributes),\n            ];\n          }\n        }\n\n        return [];\n      })\n      .filter((message): message is Message[] => message.length > 0);\n\n    // Sort the messages chronologically to reconstruct the chat history\n    const sortedMessages = [...assistantMessages, ...userMessages].sort(\n      (a, b) =>\n        new Date(a[0].createdAt).getTime() - new Date(b[0].createdAt).getTime()\n    );\n\n    return { responses, userEvents, suggestions, messages: sortedMessages };\n  };\n\n  /**\n   *\n   * @param payload\n   * @returns\n   */\n  static isSupportedEvent = async (\n    payload: SupportedEventRequest\n  ): Promise<SupportedEvent> => {\n    try {\n      const rawResponse =\n        await CommerceApiClient.getInstance().inferenceApi.v1SupportedEventPostRaw(\n          {\n            SupportedEventRequest:\n              coreSupportedEventRequestToApiRequest(payload),\n          }\n        );\n\n      // Get the actual HTTP response JSON\n      const httpResponseText = await rawResponse.raw.text();\n      const httpResponseJson = JSON.parse(httpResponseText);\n\n      return {\n        ...httpResponseJson,\n        numberOfReviews: httpResponseJson.num_of_reviews,\n        merchant_tags: httpResponseJson.merchant_tags || [],\n      };\n    } catch (err) {\n      Logger.logError(\"Failed to get response for v1SupportedEventPost\", {\n        err,\n      });\n      return {\n        supported: false,\n        ready: false,\n        category: undefined,\n        collections: [],\n        numberOfReviews: undefined,\n        top_category: undefined,\n        merchant_tags: [],\n      };\n    }\n  };\n\n  static identifyUser = async (\n    spiffyUserId: string,\n    merchantUserId: string,\n    uaDetails: ClientDetails\n  ): Promise<void> => {\n    try {\n      await CommerceApiClient.getInstance().defaultApi.v1AnalyticsIdentifyPost({\n        AnalyticsIdentifyRequest: {\n          user_id: spiffyUserId,\n          os_name: uaDetails.os,\n          os_version: uaDetails.osVersion,\n          platform: uaDetails.os,\n          device_id: uaDetails.deviceModel,\n          device_brand: uaDetails.deviceBrand,\n          device_manufacturer: uaDetails.deviceManufacturer,\n          device_model: uaDetails.deviceModel,\n          user_properties: {\n            cdp_user_id: merchantUserId,\n            browser: uaDetails.browser,\n            browser_version: uaDetails.browserVersion,\n            user_agent: uaDetails.userAgent,\n          },\n        },\n      });\n    } catch (err) {\n      Logger.logError(\"Failed to identify user\", err);\n    }\n  };\n\n  private static mapContextSourceToV1OrgConfigGetSource = (\n    source: ContextSourceEnum | undefined\n  ): V1OrgConfigGetSourceEnum | undefined => {\n    if (source === undefined) return undefined;\n    switch (source) {\n      case ContextSourceEnum.Fork:\n        return V1OrgConfigGetSourceEnum.Fork;\n      case ContextSourceEnum.Playground:\n        return V1OrgConfigGetSourceEnum.Playground;\n      case ContextSourceEnum.App:\n        return V1OrgConfigGetSourceEnum.App;\n      case ContextSourceEnum.Test:\n        return V1OrgConfigGetSourceEnum.Test;\n      default: {\n        // This ensures that if new values are added to ContextSourceEnum, we catch it\n        const exhaustiveCheck: never = source;\n        return exhaustiveCheck;\n      }\n    }\n  };\n\n  static getOrgConfig = async (\n    user_id: string\n  ): Promise<OrgConfig | undefined> => {\n    try {\n      const atomStore = getAtomStore();\n      const reactAppName = atomStore.get(reactAppNameAtom);\n      const contextSource = atomStore.get(contextSourceAtom);\n      const featureFlagService = atomStore.get(featureFlagServiceAtom); // Get featureFlagService\n      const request: V1OrgConfigGetRequest = {\n        namespace: reactAppName,\n        user_id,\n        source: this.mapContextSourceToV1OrgConfigGetSource(contextSource),\n        include_experiments: Object.values(ProductExperiment),\n        include_feature_gates: Object.entries(\n          featureFlagService?.featureFlagService?.getFeatureFlags() || {}\n        )\n          .filter(([, isEnabled]) => isEnabled)\n          .map(([featureGateName]) => featureGateName), // Use featureFlagService\n      };\n      const response =\n        await CommerceApiClient.getInstance().defaultApi.v1OrgConfigGet(\n          request\n        );\n\n      return validateOrgConfigResults(response);\n    } catch (err) {\n      Logger.logError(`Failed to get org config`, err, { err });\n      return undefined;\n    }\n  };\n\n  static addNoteToLatestConversation = async (\n    spiffyUserId: string,\n    email: string,\n    customerServiceProvider: CustomerServiceProvider\n  ) => {\n    Logger.logInfo(\n      `addNoteToLatestConversation - user_id=${spiffyUserId} email=${email} customer_service_provider=${customerServiceProvider}`\n    );\n    try {\n      await CommerceApiClient.getInstance().customerServiceApi.v1CustserviceAddNoteToLatestConversationPost(\n        {\n          AddNoteToLatestConversationRequest: {\n            spiffy_user_id: spiffyUserId,\n            email,\n            customer_service_provider: customerServiceProvider,\n          },\n        }\n      );\n    } catch (err) {\n      Logger.logError(\"Failed to add note to latest conversation\", { err });\n    }\n  };\n\n  static getCustomerServiceApi = () =>\n    CommerceApiClient.getInstance().customerServiceApi;\n}\n\nexport default CommerceApiClient;\n","import { useState, useCallback } from \"react\";\nimport {\n  SpiffyMetricsEventName,\n  useAmplitude,\n} from \"src/contexts/amplitudeContext\";\n\nexport const useAmplitudeTracking = () => {\n  const { trackEvent, isReady } = useAmplitude();\n  const [loading, setLoading] = useState(false);\n  const [error, setError] = useState<Error | null>(null);\n\n  const track = useCallback(\n    async (\n      eventName: SpiffyMetricsEventName,\n      eventProps?: Record<string, unknown>\n    ) => {\n      if (!isReady) return;\n\n      setLoading(true);\n      setError(null);\n\n      try {\n        await trackEvent({ eventName, eventProps });\n      } catch (err) {\n        setError(err instanceof Error ? err : new Error(\"Tracking failed\"));\n        throw err;\n      } finally {\n        setLoading(false);\n      }\n    },\n    [trackEvent, isReady]\n  );\n\n  return { track, loading, error, isReady };\n};\n","import { useEffect } from \"react\";\n\nexport const useBlockBackButton = (enabled: boolean, callback: () => void) => {\n\n  useEffect(() => {\n    if (enabled && window) {\n\n      if (window.history.scrollRestoration) {\n        window.history.scrollRestoration = \"manual\"\n      }\n\n      window.history.pushState(null, document.title, window.location.href);\n      window.onpopstate = (e) => {\n        e.preventDefault();\n        window.history.pushState(null, document.title, window.location.href);\n        callback?.()\n      };\n    }\n\n    return () => {\n      if (enabled && window) {\n        window.history.back()\n        window.onpopstate = null\n        window.history.scrollRestoration = \"auto\";\n      }\n    }\n  }, [enabled])\n\n}","import { useCdn } from \"../contexts/cdnContext\";\n\nexport const useCdnUrl = () => {\n  const { cdnUrl } = useCdn();\n  return cdnUrl;\n};\n\nexport const useCdnBasePath = () => {\n  const { getCdnBasePath } = useCdn();\n  return getCdnBasePath();\n};\n\nexport const useAssetUrl = (assetName: string, orgShortName: string) => {\n  const { getAssetURL } = useCdn();\n  return getAssetURL(assetName, orgShortName);\n};\n","import { useAtomValue, useSetAtom } from \"jotai\";\nimport { ChatElementDisplayLocation } from \"src/application/models/chatElementDisplayLocation\";\nimport { chatAtom, chatOnToggleAtom } from \"src/atoms/chat\";\nimport {\n  SpiffyMetricsEventName,\n  useAmplitude,\n} from \"src/contexts/amplitudeContext\";\n\nexport const useChatToggle = () => {\n  const onToggle = useSetAtom(chatOnToggleAtom);\n  const { isOpen } = useAtomValue(chatAtom);\n  const { trackEvent } = useAmplitude();\n\n  const toggle = (\n    triggerLocation: ChatElementDisplayLocation,\n    triggerId?: string\n  ) => {\n    if (!isOpen) {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.ChatComponentExpanded,\n        eventProps: {\n          message_metadata: {\n            trigger_location: triggerLocation,\n            trigger_id: triggerId,\n          },\n        },\n      });\n    } else {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.ChatComponentCollapsed,\n        eventProps: {\n          message_metadata: {\n            trigger_location: triggerLocation,\n            trigger_id: triggerId,\n          },\n        },\n      });\n    }\n\n    onToggle();\n  };\n\n  const openChat = (\n    triggerLocation: ChatElementDisplayLocation,\n    triggerId?: string\n  ) => {\n    if (!isOpen) {\n      toggle(triggerLocation, triggerId);\n    }\n  };\n\n  const closeChat = (\n    triggerLocation: ChatElementDisplayLocation,\n    triggerId?: string\n  ) => {\n    if (isOpen) {\n      toggle(triggerLocation, triggerId);\n    }\n  };\n\n  return {\n    toggle,\n    isOpen,\n    openChat,\n    closeChat,\n  };\n};\n","import { useSetAtom } from \"jotai\";\nimport { chatOnToggleAtom } from \"src/atoms/chat/chatState\";\nimport { ChatElementDisplayLocation } from \"src/application/models/chatElementDisplayLocation\";\nimport { useAmplitudeTracking } from \"./useAmplitudeOperations\";\n\nexport const useChatToggleAnalytics = () => {\n  const setChatOnToggle = useSetAtom(chatOnToggleAtom);\n  const { track } = useAmplitudeTracking();\n\n  const toggleChat = (triggerLocation?: ChatElementDisplayLocation) => {\n    setChatOnToggle(triggerLocation, track);\n  };\n\n  return { toggleChat };\n};\n","import Logger from 'src/application/logging/logger';\nimport { useCallback } from 'react';\n\n/**\n * Hook to call the `click` method of the merchant's customer support chat widget.\n *\n * @param onSwitchToAgent a function to override the function returned by the hook. This is mainly to\n * preserve backward compatibility for merchants not using Kustomer and will be removed when all\n * CS integrations are handled.\n *\n * @returns a function that searches for the customer support chat widget and calls the `click` method.\n */\nexport const useCustomerSupportHandoff = (onSwitchToAgent?: () => void) => {\n  // TODO handle Gorgias\n\n  const onKustomerSwitch = useCallback(() => {\n    const kustomerElement = document.getElementById('kustomer-ui-sdk-iframe');\n\n    if (kustomerElement == null || !(kustomerElement instanceof HTMLIFrameElement)) {\n      Logger.logError('[spiffy-ai] Kustomer iFrame element not found', undefined);\n      return;\n    }\n\n    const kustomerButton = kustomerElement.contentWindow?.document?.getElementById('rootChatIcon');\n\n    if (kustomerButton == null) {\n      Logger.logError('[spiffy-ai] Kustomer button not found', undefined);\n      return;\n    }\n\n    kustomerButton.click();\n  }, []);\n\n  if (onSwitchToAgent != null) {\n    return { onSwitch: onSwitchToAgent };\n  }\n\n  return { onSwitch: onKustomerSwitch };\n};\n","import { useState, useEffect } from 'react';\n\nexport function useDebounce<T>(value: T, delay: number): T {\n  const [debouncedValue, setDebouncedValue] = useState<T>(value);\n\n  useEffect(() => {\n    const handler = setTimeout(() => {\n      setDebouncedValue(value);\n    }, delay);\n\n    return () => {\n      clearTimeout(handler);\n    };\n  }, [value, delay]);\n\n  return debouncedValue;\n}\n","import { ReactElement, useEffect, useRef, useState } from 'react';\nimport { DOMObserver } from 'src/application/utils/domObserver';\nimport { ElementObserver } from 'src/application/utils/elementObserver';\nimport { MouseEventTypes } from 'src/application/utils/mouseEventTypes';\nimport { NodeSelector } from 'src/application/utils/nodeSelector';\n\nexport interface ElementObserverUtility {\n  onChange: (fn: (el: HTMLElement | null) => void) => void;\n  onAdd: (fn: (el: HTMLElement | null) => void) => void;\n  onRemove: (fn: (el: HTMLElement | null) => void) => void;\n  onClassChange: (fn: (classes?: DOMTokenList) => void) => void;\n  onClassAdded: (className: string, fn: () => void) => void;\n  onClassRemoved: (className: string, fn: () => void) => void;\n  onAddChild: (fn: (nodes?: DOMTokenList) => void) => void;\n  onRemoveChild: (fn: (nodes?: DOMTokenList) => void) => void;\n  onEvent: <K extends keyof HTMLElementEventMap>(\n    event: K,\n    fn: EventListenerOrEventListenerObject,\n  ) => void;\n  blockRendering: () => void;\n  unblockRendering: () => void;\n  exists: () => boolean;\n  isRendered: () => boolean;\n  render: (fn: () => ReactElement) => ReactElement | undefined;\n  fire: (event: MouseEventTypes) => void;\n  show: () => void;\n  hide: () => void;\n  applyStyle: (styles: React.CSSProperties) => void;\n  targetNode: HTMLElement | null;\n}\n\n/*\n * This hook enables the connection between a `MutationObserver` and a React.js component.\n * It abstracts all the necessary validations and implementations to modify HTML elements,\n * monitor their changes, and interact with them. The only required input is the element's selector.\n *\n * How to use it:\n *\n * Let's assume we want to interact with the following element:\n * `<div id=\"awesome-content\"><content /></div>`\n *\n * To hook into that element, simply do the following:\n * `const element = useElementObserver(SelectorFactory.id(\"awesome-content\"))`\n *\n * With the `ElementObserver` instance in hand, we can interact with the HTML element without having\n * to manage its presence in the DOM. The `onAdd` and `onRemove` functions can be used to determine\n * whether the element is present in the DOM or not, and to ensure interaction occurs only when necessary.\n *\n * @param selector\n * @returns\n */\nexport const useElementObserver = (selector: NodeSelector): ElementObserverUtility => {\n  const INITIAL_RENDER_STATE = true;\n  const eoRef = useRef<ElementObserver>(DOMObserver.add(selector));\n  const [renderBlocked, setRenderBlocked] = useState(INITIAL_RENDER_STATE);\n\n  /**\n   * Fired every time the HTML element changes.\n   *\n   * @param fn\n   */\n  const onChange = (fn: (el: HTMLElement | null) => void) => {\n    eoRef.current?.registerOnChange(fn);\n  };\n\n  /**\n   * Fired when the HTML element is added to the DOM.\n   *\n   * @param fn\n   */\n  const onAdd = (fn: (el: HTMLElement | null) => void) => {\n    eoRef.current?.registerOnAdd(fn);\n  };\n\n  /**\n   * Fired when the HTML element is removed from the DOM.\n   *\n   * @param fn\n   */\n  const onRemove = (fn: (el: HTMLElement | null) => void) => {\n    eoRef.current?.registerOnRemove(fn);\n  };\n\n  /**\n   * Fired when the class of the HTML element changes.\n   *\n   * @param fn\n   */\n  const onClassChange = (fn: (classes?: DOMTokenList) => void) => {\n    eoRef.current?.registerOnclassChange(fn);\n  };\n\n  /**\n   * Fired when a class is added to the HTML element.\n   *\n   * @param className\n   * @param fn\n   */\n  const onClassAdded = (className: string, fn: () => void) => {\n    eoRef.current?.registerOnClassAdded(className, fn);\n  };\n\n  /**\n   * Fired when a class is removed from the HTML element.\n   *\n   * @param className\n   * @param fn\n   */\n  const onClassRemoved = (className: string, fn: () => void) => {\n    eoRef.current?.registerOnClassRemoved(className, fn);\n  };\n\n  /**\n   * Fired when a child element is added to the HTML element.\n   *\n   * @param fn\n   */\n  const onAddChild = (fn: (nodes?: DOMTokenList) => void) => {\n    eoRef.current?.registerOnAddChild(fn);\n  };\n\n  /**\n   * Fired when a child element is removed from the HTML element.\n   *\n   * @param fn\n   */\n  const onRemoveChild = (fn: (nodes?: DOMTokenList) => void) => {\n    eoRef.current?.registerOnRemoveChild(fn);\n  };\n\n  /**\n   * Allows hooking event listeners to the HTML element, such as `focus`, `blur`, etc.\n   *\n   * @param event\n   * @param fn\n   */\n  const onEvent = <K extends keyof HTMLElementEventMap>(\n    event: K,\n    fn: EventListenerOrEventListenerObject,\n  ) => {\n    eoRef.current.registerEvent(event, fn);\n  };\n\n  /**\n   * Useful when rendering a React.js component inside the HTML element.\n   *\n   * @param fn\n   * @returns\n   */\n  const render = (fn: () => ReactElement) => {\n    if (!renderBlocked) {\n      return eoRef.current.render(fn);\n    }\n  };\n\n  /**\n   * Checks if the element exists in the DOM.\n   *\n   * @returns\n   */\n  const exists = () => !!eoRef.current.getNode();\n\n  /**\n   * Checks if rendering is unblocked.\n   *\n   * @returns\n   */\n  const isRendered = () => !renderBlocked;\n\n  /**\n   * Triggers an event for the HTML element.\n   *\n   * @param event\n   */\n  const fire = (event: MouseEventTypes) => {\n    eoRef.current.fire(event);\n  };\n\n  /**\n   * Shows the HTML element.\n   *\n   * @returns\n   */\n  const show = () => eoRef.current.show();\n\n  /**\n   * Hides the HTML element.\n   *\n   * @returns\n   */\n  const hide = () => eoRef.current.hide();\n\n  /**\n   * Blocks the rendering of elements.\n   *\n   * @returns\n   */\n  const blockRendering = () => setRenderBlocked(true);\n\n  /**\n   * Unblocks the rendering of elements.\n   *\n   * @returns\n   */\n  const unblockRendering = () => setRenderBlocked(false);\n\n  /**\n   * Applies CSS styles to the HTML element.\n   *\n   * @param styles\n   */\n  const applyStyle = (styles: React.CSSProperties) => {\n    const node = eoRef?.current?.getNode();\n    node && Object.assign(node.style, styles);\n  };\n\n  useEffect(() => {\n    eoRef.current.init();\n    eoRef.current.registerOnReset(() => setRenderBlocked(INITIAL_RENDER_STATE));\n    DOMObserver.observe();\n    return () => DOMObserver.remove(selector);\n  }, [selector.getPattern()]);\n\n  return {\n    targetNode: eoRef.current.getNode(),\n    onChange,\n    onAdd,\n    onRemove,\n    onClassChange,\n    onClassAdded,\n    onClassRemoved,\n    onAddChild,\n    onRemoveChild,\n    onEvent,\n    blockRendering,\n    unblockRendering,\n    exists,\n    isRendered,\n    render,\n    fire,\n    show,\n    hide,\n    applyStyle,\n  } satisfies ElementObserverUtility;\n};\n","import { useRef, useState } from \"react\";\n\ntype ArrowPosition = 'lt' | 'ct' | 'rt'\n\ninterface AnimationProps {\n    element: HTMLElement\n    targetScroll: number \n    duration: number \n    direction?: 'lt' | 'rt' \n    multiply?: number \n    offset?: number\n    callback?: (position: ArrowPosition) => void\n}\n\nconst animateHorizontalScroll = ({ element, duration, targetScroll, multiply = 1, direction, callback, offset = 0 }: AnimationProps) => {\n    const start = element.scrollLeft\n    const distance = (targetScroll - start) * multiply;\n    const startTime = performance.now();\n\n    function easeOutSine(x: number): number {\n      return Math.sin((x * Math.PI) / 2);\n    }\n\n    function scrollStep(currentTime: number): void {\n        const timeElapsed = currentTime - startTime;\n        const progress = Math.min(timeElapsed / duration, 1);\n        const easing = easeOutSine(progress);\n        const step = start + (distance * easing)\n        const canScroll = (direction === 'rt' ? (element.scrollLeft < step) : (element.scrollLeft > step)) || !direction\n        \n        if (step > 0 && canScroll) {\n          element.scrollTo(step, 0)\n        }\n\n        if (timeElapsed < duration) {\n            requestAnimationFrame(scrollStep);\n        } \n\n        // End of scrolling container\n        else if ((element.scrollLeft + offset) === element.scrollWidth) {\n            callback?.('rt')\n        } \n        // Begin of scrolling container\n        else if (element.scrollLeft <= 1) {\n            callback?.('lt')\n        }\n        // Somewhere in the middle\n        else {\n            callback?.('ct')\n        }\n\n    }\n\n    requestAnimationFrame(scrollStep);\n}\n\n\nexport const useGrabAndScroll = (enabled: boolean, chunkWidth: number, speed: number = 400, offset: number = 0) => {\n    const containerRef = useRef<HTMLDivElement>(null)\n    const [ leftArrow, setLeftArrow ] = useState(false)\n    const [ rightArrow, setRightArrow ] = useState(true)\n\n    const handleArrows = (position: ArrowPosition) => {\n        switch (position) {\n            case 'lt':\n                setLeftArrow(false)\n                setRightArrow(true)\n                break\n\n            case 'rt':\n                setLeftArrow(true)\n                setRightArrow(false)\n                break\n\n            default:\n                setLeftArrow(true)\n                setRightArrow(true)\n        }\n    }\n\n    const animationTrigger = () => {\n        if (enabled && containerRef.current) {\n            const dist = (containerRef?.current?.scrollLeft || 0) / chunkWidth\n            const targetScroll = chunkWidth * (Math.floor(dist) + (dist % 1 > 0.5 ? 1 : 0))\n            animateHorizontalScroll({\n                element: containerRef.current, \n                targetScroll, \n                duration: speed,\n                offset,\n                callback: handleArrows\n            })\n        }\n    }\n\n    const onNext = (cardsToSlide: number) => {\n        if (containerRef.current) {\n            const targetScroll = containerRef.current.scrollLeft + chunkWidth\n            animateHorizontalScroll({\n                element: containerRef.current, \n                targetScroll, \n                duration: speed, \n                direction: 'rt', \n                multiply: cardsToSlide,\n                offset,\n                callback: handleArrows\n            })\n        }\n    }\n    \n    const onPrevious = (cardsToSlide: number) => {\n        if (containerRef.current) {\n            const targetScroll = containerRef.current.scrollLeft - chunkWidth\n            animateHorizontalScroll({\n                element: containerRef.current, \n                targetScroll, \n                duration: speed, \n                direction: 'lt', \n                multiply: cardsToSlide,\n                offset,\n                callback: handleArrows\n            })\n        }\n    }\n\n    return {\n        containerRef,\n        leftArrow,\n        rightArrow,\n        animationTrigger,\n        onNext,\n        onPrevious\n    }\n}","import React, {\n  createContext,\n  useContext,\n  useCallback,\n  useMemo,\n  ReactNode,\n} from \"react\";\nimport { useAtomValue } from \"jotai\";\nimport { baseUrlAtom, orgLevelApiKeyAtom } from \"src/atoms/envive/enviveConfig\";\nimport { getMerchantOrgIdQuery } from \"src/application/models/graphql/queries/getMerchantOrgIdQuery\";\nimport { validateGraphQLOrgId } from \"src/application/models/validators/validateGraphQLOrgId\";\nimport {\n  getMerchantColorsQuery,\n  GetMerchantColorsQueryData,\n  getMerchantFrontendConfigQuery,\n  GetMerchantFrontendConfigQueryData,\n} from \"src/application/models/graphql\";\nimport { validateGraphQLColorsConfig } from \"src/application/models/validators/validateGraphQLColorsConfig\";\nimport { validateGraphQLFrontendConfig } from \"src/application/models/validators/validateGraphQLFrontendConfig\";\nimport { ColorMapping } from \"src/application/models/colorsConfig\";\nimport { FrontendConfig } from \"src/application/models/frontendConfig\";\nimport Logger from \"src/application/logging/logger\";\nimport { configVersion } from \"src/types/config-versions\";\nimport {\n  CamelCasedPropertiesDeep,\n  transformSnakeToCamel,\n} from \"src/application/models\";\n\ninterface BaseMeConfigQueryResponse {\n  me: {\n    org?: {\n      id: string;\n    };\n    getProductsConfigByVersion?: {\n      frontend?: {\n        values: CamelCasedPropertiesDeep<GetMerchantFrontendConfigQueryData>;\n      };\n      colors?: { values: CamelCasedPropertiesDeep<GetMerchantColorsQueryData> };\n    };\n  };\n}\nexport type ColorsConfigResponse = CamelCasedPropertiesDeep<ColorMapping>;\nexport type FrontendConfigResponse = CamelCasedPropertiesDeep<FrontendConfig>;\n\nexport type GraphQlConfigValues = {\n  colorsConfig?: ColorsConfigResponse;\n  frontendConfig?: CamelCasedPropertiesDeep<FrontendConfig>;\n};\n\ninterface GraphQLContextValue {\n  executeQuery: <T>(\n    query: string,\n    variables?: Record<string, unknown>\n  ) => Promise<T>;\n  getOrgId: () => Promise<string | undefined>;\n  getColorsAndFrontendConfig: () => Promise<GraphQlConfigValues>;\n  isReady: boolean;\n}\n\nconst GraphQLContext = createContext<GraphQLContextValue | null>(null);\n\nconst colorsAndFrontendConfigQuery = () => `\n  query ($version: String = \"${configVersion()}\") {\n    me {\n      getProductsConfigByVersion(version: $version) {\n        frontend { values }\n        colors { values }\n      }\n    }\n  }\n`;\n\ntype GraphQLProviderProps = {\n  children: ReactNode;\n};\n\nexport const GraphQLProvider = ({ children }: GraphQLProviderProps) => {\n  const apiKey = useAtomValue(orgLevelApiKeyAtom);\n  const baseUrl = useAtomValue(baseUrlAtom);\n\n  const isReady = Boolean(apiKey && baseUrl);\n\n  const executeQuery = useCallback(\n    async (query: string, variables?: Record<string, unknown>) => {\n      if (!isReady) {\n        throw new Error(\"GraphQL client not ready - missing apiKey or baseUrl\");\n      }\n\n      const response = await fetch(`${baseUrl}/v1/graphql`, {\n        method: \"POST\",\n        headers: {\n          \"Content-Type\": \"application/json\",\n          Authorization: `Bearer ${apiKey}`,\n        },\n        body: JSON.stringify({ query, variables }),\n      });\n\n      if (!response.ok) {\n        throw new Error(`GraphQL request failed: ${response.statusText}`);\n      }\n\n      const result = await response.json();\n      if (result.errors) {\n        throw new Error(`GraphQL errors: ${JSON.stringify(result.errors)}`);\n      }\n\n      return result.data;\n    },\n    [apiKey, baseUrl, isReady]\n  );\n\n  const getOrgId = useCallback(async () => {\n    const response = await executeQuery(getMerchantOrgIdQuery);\n    return validateGraphQLOrgId(response.me.org?.id);\n  }, [executeQuery]);\n\n  const getColorsAndFrontendConfig =\n    useCallback(async (): Promise<GraphQlConfigValues> => {\n      try {\n        const query = await colorsAndFrontendConfigQuery();\n        if (!query) {\n          throw new Error(\"Colors and frontend config query is not defined\");\n        }\n        const response = await executeQuery(query);\n        const colorsConfig =\n          response.me.getProductsConfigByVersion?.colors?.values;\n        const frontendConfig =\n          response.me.getProductsConfigByVersion?.frontend?.values;\n        const transformedColorConfig = transformSnakeToCamel(colorsConfig);\n        const transformedFrontendConfig = transformSnakeToCamel(frontendConfig);\n        return {\n          colorsConfig: transformedColorConfig as ColorsConfigResponse,\n          frontendConfig: transformedFrontendConfig as FrontendConfigResponse,\n        };\n      } catch (err) {\n        Logger.logError(\n          \"Error fetching graphql colors and frontend config\",\n          err\n        );\n        return { colorsConfig: undefined, frontendConfig: undefined };\n      }\n    }, [executeQuery]);\n\n  const value = useMemo(\n    () => ({\n      executeQuery,\n      getOrgId,\n      getColorsAndFrontendConfig,\n      isReady,\n    }),\n    [executeQuery, getOrgId, getColorsAndFrontendConfig, isReady]\n  );\n\n  return (\n    <GraphQLContext.Provider value={value}>{children}</GraphQLContext.Provider>\n  );\n};\n\nexport const useGraphQLClient = () => {\n  const context = useContext(GraphQLContext);\n  if (!context) {\n    throw new Error(\"useGraphQLClient must be used within a GraphQLProvider\");\n  }\n  return context;\n};\n","import { useCallback, useEffect, useState } from \"react\";\nimport { useGraphQLClient } from \"src/contexts/graphqlContext\";\nimport { ColorMapping } from \"src/application/models/colorsConfig\";\nimport { FrontendConfig } from \"src/application/models/frontendConfig\";\nimport { transformSnakeToCamel } from \"src/application/models\";\n\nexport const useColorsAndFrontendConfig = () => {\n  const { getColorsAndFrontendConfig, isReady } = useGraphQLClient();\n  const [data, setData] = useState({});\n  const [loading, setLoading] = useState(false);\n  const [error, setError] = useState<Error | null>(null);\n\n  const fetchConfig = useCallback(async () => {\n    if (!isReady) return;\n\n    setLoading(true);\n    setError(null);\n\n    try {\n      const result = await getColorsAndFrontendConfig();\n      setData(result);\n    } catch (err) {\n      setError(err instanceof Error ? err : new Error(\"Unknown error\"));\n    } finally {\n      setLoading(false);\n    }\n  }, [getColorsAndFrontendConfig, isReady]);\n\n  useEffect(() => {\n    fetchConfig();\n  }, [fetchConfig]);\n\n  return { data, loading, error, refetch: fetchConfig };\n};\n\nexport const useOrgId = () => {\n  const { getOrgId, isReady } = useGraphQLClient();\n  const [orgId, setOrgId] = useState<string | undefined>();\n  const [loading, setLoading] = useState(false);\n  const [error, setError] = useState<Error | null>(null);\n\n  useEffect(() => {\n    if (!isReady) return;\n\n    const fetchOrgId = async () => {\n      setLoading(true);\n      setError(null);\n\n      try {\n        const id = await getOrgId();\n        setOrgId(id);\n      } catch (err) {\n        setError(err instanceof Error ? err : new Error(\"Unknown error\"));\n      } finally {\n        setLoading(false);\n      }\n    };\n\n    fetchOrgId();\n  }, [getOrgId, isReady]);\n\n  return { orgId, loading, error };\n};\n","import React, {\n  createContext,\n  useCallback,\n  useContext,\n  useMemo,\n  useState,\n  useEffect,\n} from \"react\";\nimport UAParser from \"ua-parser-js\";\nimport Logger from \"src/application/logging/logger\";\nimport CommerceApiClient from \"src/application/commerce-api\";\nimport { v4 as uuid } from \"uuid\";\nimport { ClientDetails } from \"src/application/models/clientDetails\";\nimport { useLocalStorage } from \"src/contexts/localStorageContext\";\n\n// Helper function from the original service\nconst getUserAgentDetails = (): ClientDetails => {\n  const uaParser = new UAParser();\n  const result = uaParser.getResult();\n\n  return {\n    os: result?.os?.name,\n    osVersion: result?.os?.version,\n    deviceBrand: result?.device?.vendor,\n    deviceManufacturer: result?.device?.vendor,\n    deviceModel: result?.device?.model,\n    browser: result?.browser?.name,\n    browserVersion: result?.browser?.version,\n    userAgent: result?.ua,\n  };\n};\n\nexport interface UserIdentityContextType {\n  identifyUser: () => Promise<void>;\n  getUserIdOrDefault: () => string;\n  getUserIdOverrideFromLocalStorage: () => string | undefined;\n  getUserIdDefaultFromLocalStorage: () => string | undefined;\n  setUserIdDefaultInLocalStorage: (userId: string) => string;\n  setUserIdOverrideInLocalStorage: (userId: string) => string;\n  clearUserIdOverrideInLocalStorage: () => void;\n  isReady: boolean;\n}\n\nconst UserIdentityContext = createContext<UserIdentityContextType | undefined>(\n  undefined\n);\n\nexport const UserIdentityProvider: React.FC<{ children: React.ReactNode }> = ({\n  children,\n}) => {\n  const {\n    getItem,\n    setItem,\n    isAvailable: localStorageIsReady,\n  } = useLocalStorage();\n\n  const [isReady, setIsReady] = useState(false);\n\n  useEffect(() => {\n    // Assuming CommerceApiClient and other dependencies are ready if localStorage is.\n    // In a more complex scenario, you might have more checks here.\n    setIsReady(localStorageIsReady);\n  }, [localStorageIsReady]);\n\n  const USER_ID_OVERRIDE_KEY = \"v1-spiffy-user-id-override\";\n  const USER_ID_DEFAULT_KEY = \"v1-spiffy-user-id-default\";\n\n  const getUserIdOverrideFromLocalStorage = useCallback(():\n    | string\n    | undefined => {\n    return getItem(USER_ID_OVERRIDE_KEY) ?? undefined;\n  }, [getItem]);\n\n  const getUserIdDefaultFromLocalStorage = useCallback(():\n    | string\n    | undefined => {\n    return getItem(USER_ID_DEFAULT_KEY) ?? undefined;\n  }, [getItem]);\n\n  const setUserIdDefaultInLocalStorage = useCallback(\n    (userId: string): string => {\n      Logger.logInfo(\n        `setUserIdDefaultInLocalStorage - Setting user_id=${userId}`\n      );\n      setItem(USER_ID_DEFAULT_KEY, userId);\n      // window.dispatchEvent is handled by useLocalStorage now\n      return userId;\n    },\n    [setItem, USER_ID_DEFAULT_KEY]\n  );\n\n  const setUserIdOverrideInLocalStorage = useCallback(\n    (userId: string): string => {\n      Logger.logInfo(\n        `setUserIdOverrideInLocalStorage - Setting user_id=${userId}`\n      );\n      setItem(USER_ID_OVERRIDE_KEY, userId);\n      // window.dispatchEvent is handled by useLocalStorage now\n      return userId;\n    },\n    [setItem, USER_ID_OVERRIDE_KEY]\n  );\n\n  const clearUserIdOverrideInLocalStorage = useCallback(() => {\n    Logger.logInfo(`clearUserIdOverrideInLocalStorage - Clearing user_id`);\n    // LocalStorageService.getLocalStorage()?.removeItem(USER_ID_OVERRIDE_KEY);\n    // window.dispatchEvent is handled by useLocalStorage now\n    setItem(USER_ID_OVERRIDE_KEY, \"\"); // Set to empty string to clear\n  }, [setItem, USER_ID_OVERRIDE_KEY]);\n\n  const getUserIdOrDefault = useCallback((): string => {\n    const userIdOverride = getUserIdOverrideFromLocalStorage();\n    if (userIdOverride) {\n      return userIdOverride;\n    }\n\n    const defaultUserId = getUserIdDefaultFromLocalStorage();\n    if (defaultUserId) {\n      return defaultUserId;\n    }\n\n    return setUserIdDefaultInLocalStorage(`spiffy-user-id-${uuid()}`);\n  }, [\n    getUserIdOverrideFromLocalStorage,\n    getUserIdDefaultFromLocalStorage,\n    setUserIdDefaultInLocalStorage,\n  ]);\n\n  const identifyUser = useCallback(async (): Promise<void> => {\n    if (!isReady) {\n      Logger.logWarn(\n        \"[UserIdentityContext] Context not ready, skipping identifyUser\",\n        undefined\n      );\n      return;\n    }\n\n    try {\n      // Temporarily commented out until WindowDataLayerService is resolved\n      // const cdpUserId = WindowDataLayerService.getGoogleAnalyticsClientId();\n      const cdpUserId = \"UNKNOWN_CDP_USER_ID\"; // Placeholder\n      const userId = getUserIdOrDefault();\n      const userAgentDetails = getUserAgentDetails();\n\n      if (!cdpUserId) {\n        Logger.logWarn(\n          \"[spiffy-ai] No GA Client ID found, skipping identifyUser\",\n          undefined\n        );\n        return;\n      }\n\n      await CommerceApiClient.identifyUser(userId, cdpUserId, userAgentDetails);\n    } catch (error) {\n      Logger.logError(\"[spiffy-ai] Error identifying user\", error);\n    }\n  }, [isReady, getUserIdOrDefault]);\n\n  const value = useMemo(\n    () => ({\n      identifyUser,\n      getUserIdOrDefault,\n      getUserIdOverrideFromLocalStorage,\n      getUserIdDefaultFromLocalStorage,\n      setUserIdDefaultInLocalStorage,\n      setUserIdOverrideInLocalStorage,\n      clearUserIdOverrideInLocalStorage,\n      isReady,\n    }),\n    [\n      identifyUser,\n      getUserIdOrDefault,\n      getUserIdOverrideFromLocalStorage,\n      getUserIdDefaultFromLocalStorage,\n      setUserIdDefaultInLocalStorage,\n      setUserIdOverrideInLocalStorage,\n      clearUserIdOverrideInLocalStorage,\n      isReady,\n    ]\n  );\n\n  return (\n    <UserIdentityContext.Provider value={value}>\n      {children}\n    </UserIdentityContext.Provider>\n  );\n};\n\nexport const useUserIdentity = () => {\n  const context = useContext(UserIdentityContext);\n  if (!context) {\n    throw new Error(\n      \"useUserIdentity must be used within a UserIdentityProvider\"\n    );\n  }\n  return context;\n};\n","import { useState, useCallback } from \"react\";\nimport { useUserIdentity } from \"src/contexts/userIdentityContext\";\n\nexport const useIdentifyUser = () => {\n  const { identifyUser, isReady } = useUserIdentity();\n  const [loading, setLoading] = useState(false);\n  const [error, setError] = useState<Error | null>(null);\n\n  const executeIdentifyUser = useCallback(async () => {\n    if (!isReady) {\n      setError(new Error(\"UserIdentityContext not ready.\"));\n      return;\n    }\n\n    setLoading(true);\n    setError(null);\n\n    try {\n      await identifyUser();\n    } catch (err) {\n      setError(\n        err instanceof Error\n          ? err\n          : new Error(\"Unknown error during user identification.\")\n      );\n      throw err;\n    } finally {\n      setLoading(false);\n    }\n  }, [identifyUser, isReady]);\n\n  return { loading, error, executeIdentifyUser, isReady };\n};\n","import { useAtomValue } from \"jotai\";\nimport { OrgShortName } from \"src/application/models\";\nimport { orgShortNameAtom } from \"src/atoms/envive/enviveConfig\";\nabstract class ImageResolver {\n  abstract resolve(url: string, size: number): string;\n}\n\nclass MerchantImageResolver {\n  private static imageResolverMap = new Map<string, ImageResolver>();\n\n  private static loadMapping() {\n    if (this.imageResolverMap.size === 0) {\n      this.imageResolverMap.set(OrgShortName.Spanx, new ShopifyImageResolver());\n      this.imageResolverMap.set(\n        OrgShortName.SpanxStaging,\n        new ShopifyImageResolver()\n      );\n      this.imageResolverMap.set(\n        OrgShortName.UniqueVintage,\n        new ShopifyImageResolver()\n      );\n    }\n    return this.imageResolverMap;\n  }\n\n  static get(name: string) {\n    return this.loadMapping().get(name);\n  }\n}\n\nclass ShopifyImageResolver extends ImageResolver {\n  resolve(url: string, size: number): string {\n    const pattern = /_\\d+x\\.jpg/;\n    const urlHasPrefix = pattern.test(url);\n    const newSizePrefix = `_${size}x.jpg`;\n    if (urlHasPrefix) {\n      return url.replace(pattern, newSizePrefix);\n    }\n    return url.replace(\".jpg\", newSizePrefix);\n  }\n}\n\nexport const useImageResolver = () => {\n  const orgShortName = useAtomValue(orgShortNameAtom);\n  const resolve = (image?: string, size?: number) => {\n    if (image && size && orgShortName) {\n      const newImagePath = MerchantImageResolver.get(orgShortName)?.resolve(\n        image,\n        size\n      );\n      return newImagePath || image;\n    }\n    return image;\n  };\n\n  return {\n    resolve,\n  };\n};\n","import { RefObject, useEffect, useState } from 'react';\n\n// https://rasilbaidar.medium.com/trigger-event-when-element-enters-viewport-the-react-way-168509da2e23\nexport const useIntersection = (element: RefObject<HTMLElement>, rootMargin: string) => {\n  const [isVisible, setIsVisible] = useState(false);\n\n  useEffect(() => {\n    const current = element?.current;\n    const observer = new IntersectionObserver(\n      ([entry]) => {\n        setIsVisible(entry.isIntersecting);\n      },\n      { rootMargin },\n    );\n\n    if (current) {\n      observer?.observe(current);\n    }\n\n    return () => {\n      if (current) {\n        observer.unobserve(current);\n      }\n    };\n  }, []);\n\n  return isVisible;\n};\n","import { useState, useEffect } from 'react';\n\nexport const useIsSmallScreen = () => {\n  const [isSmall, setIsSmall] = useState(false);\n\n  useEffect(() => {\n    const mediaQuery = window.matchMedia('(max-width: 479px)');\n\n    // Set initial value\n    setIsSmall(mediaQuery.matches);\n\n    // Update state when viewport changes\n    const handleResize = (event: MediaQueryListEvent) => {\n      setIsSmall(event.matches);\n    };\n\n    mediaQuery.addEventListener('change', handleResize);\n\n    return () => mediaQuery.removeEventListener('change', handleResize);\n  }, []);\n\n  return isSmall;\n};\n","import { useState, useEffect, useCallback } from \"react\";\nimport {\n  useLocalStorage,\n  LocalStorageKeys,\n} from \"../contexts/localStorageContext\";\nimport { LocalStorageEventListener } from \"../application/models/localStorageEventListener\";\n\n// Reactive localStorage value hook\nexport const useLocalStorageValue = (key: string) => {\n  const { getItem, setItem, attachListener, detachListener } =\n    useLocalStorage();\n  const [value, setValue] = useState<string | null>(() => getItem(key));\n\n  useEffect(() => {\n    const listener: LocalStorageEventListener = {\n      storageKey: key,\n      listener: (event: StorageEvent) => {\n        setValue(event.newValue);\n      },\n    };\n\n    attachListener(listener);\n    return () => detachListener(listener);\n  }, [key, attachListener, detachListener]);\n\n  const updateValue = useCallback(\n    (newValue: string) => {\n      setItem(key, newValue);\n      setValue(newValue);\n    },\n    [key, setItem]\n  );\n\n  return { value, setValue: updateValue };\n};\n\n// Feature flag hook for Spiffy\nexport const useSpiffyFeatureFlag = () => {\n  const { setSpiffyOnFeatureFlag } = useLocalStorage();\n  const { value } = useLocalStorageValue(LocalStorageKeys.SpiffyOnOverride);\n\n  const setFlag = useCallback(\n    (flag: boolean | null) => {\n      setSpiffyOnFeatureFlag(flag);\n    },\n    [setSpiffyOnFeatureFlag]\n  );\n\n  return {\n    value: value === \"true\" ? true : value === \"false\" ? false : null,\n    setFlag,\n  };\n};\n\n// Feature flag hook for Envive\nexport const useEnviveFeatureFlag = () => {\n  const { setItem, getItem } = useLocalStorage();\n  const { value } = useLocalStorageValue(LocalStorageKeys.EnviveOnOverride);\n\n  const setFlag = useCallback(\n    (flag: boolean | null) => {\n      if (flag === true) {\n        setItem(LocalStorageKeys.EnviveOnOverride, \"true\");\n      } else if (flag === false) {\n        setItem(LocalStorageKeys.EnviveOnOverride, \"false\");\n      }\n    },\n    [setItem]\n  );\n\n  return {\n    value: value === \"true\" ? true : value === \"false\" ? false : null,\n    setFlag,\n  };\n};\n\n// Storage listener hook\nexport const useLocalStorageListener = (\n  key: string,\n  callback: (event: StorageEvent) => void\n) => {\n  const { attachListener, detachListener } = useLocalStorage();\n\n  useEffect(() => {\n    const listener: LocalStorageEventListener = {\n      storageKey: key,\n      listener: callback,\n    };\n    attachListener(listener);\n    return () => detachListener(listener);\n  }, [key, callback, attachListener, detachListener]);\n};\n","import { Message, MessageRole, MessageType } from 'src/application/models';\n\ntype MessageFinder = {\n  msgs: Message[][];\n  type?: MessageType;\n  role?: MessageRole;\n};\n\nexport const useMessageFilter = () => {\n  const findMessageIndex = ({ msgs, type, role }: MessageFinder) => {\n    let lastIndex = -1;\n    msgs.forEach((subArray, index) => {\n      subArray.forEach((obj) => {\n        if (obj.type === type || obj.role === role) lastIndex = index;\n      });\n    });\n    return lastIndex;\n  };\n\n  const removePreviousDiscussions = (msgs: Message[][], index: number) => {\n    if (index > -1) {\n      const lastMessages = msgs.slice(index);\n      return lastMessages.length > 0 ? lastMessages : msgs;\n    }\n    return msgs;\n  };\n\n  const getFilteredMessages = (msgs: Message[][], skipFilter?: boolean) => {\n    const messageMap = msgs.reduce(\n      (acc, msg) => {\n        acc[msg[0].id] = msg;\n        return acc;\n      },\n      {} as Record<string, Message[]>,\n    );\n    const deduplicatedMsgs = Object.values(messageMap);\n    if (!skipFilter) {\n      const idx = findMessageIndex({ msgs: deduplicatedMsgs, type: MessageType.Separator });\n      return removePreviousDiscussions(deduplicatedMsgs, idx);\n    }\n    return deduplicatedMsgs;\n  };\n\n  return {\n    findMessageIndex,\n    removePreviousDiscussions,\n    getFilteredMessages,\n  };\n};\n","import { useEffect } from \"react\";\n\nexport const useMessageScrollObserver = (\n    boxRef: React.RefObject<HTMLDivElement>, \n    scrollRef: React.RefObject<HTMLDivElement>, \n    onScrollChange: (position: number) => void\n) => {\n\n    const calculateScrollHeight = () => {\n      const boxHeight = boxRef?.current?.getBoundingClientRect().height || 0\n      const scrollHeight = scrollRef?.current?.getBoundingClientRect().height || 0\n      return boxHeight - scrollHeight\n    }\n\n    const updateState = () => {\n      const scrollHeight = calculateScrollHeight()\n      if (scrollHeight > 0) {\n        onScrollChange(scrollHeight)\n      }\n    }\n\n    useEffect(() => {\n      let boxRO = null;\n      let scrollRO = null;\n\n      if (scrollRef?.current) {\n        boxRO = new ResizeObserver(updateState);\n        boxRO.observe(scrollRef?.current);\n      }\n      \n      if (boxRef?.current) {\n        scrollRO = new ResizeObserver(updateState);\n        scrollRO.observe(boxRef?.current);\n      }\n\n      return () => {\n        if (scrollRef?.current && boxRO) {\n          boxRO.unobserve(scrollRef?.current);\n        }\n\n        if (scrollRO && boxRef?.current) {\n          scrollRO?.unobserve(boxRef?.current)\n        }\n      };\n    }, []);\n\n}","import React, { createContext, useContext, useMemo, ReactNode } from \"react\";\nimport { OrgConfigFeatureGate } from \"src/application/models/api/orgConfigResults\";\nimport { FeatureGates } from \"src/application/models/featureGates\";\nimport Logger from \"src/application/logging/logger\";\n\n// This is the class that was previously implicitly used or defined elsewhere\nclass FeatureFlagService {\n  private featureGates: OrgConfigFeatureGate[];\n\n  constructor(featureGates: OrgConfigFeatureGate[]) {\n    this.featureGates = featureGates;\n  }\n\n  isFeatureGateEnabled = (featureGate: FeatureGates): boolean => {\n    const gateValue = this.featureGates.find(\n      (gate) => gate.name === featureGate\n    );\n\n    // TODO: Add logic for overrides (query params, window, stored) if needed, similar to the old FeatureFlagContext.tsx\n    // For now, direct value from config is used.\n\n    if (gateValue == null || gateValue.value == null) {\n      Logger.logDebug(\n        `[spiffy-ai] isFeatureGateEnabled featureGate:${featureGate} value is undefined - returning false`\n      );\n      return false;\n    }\n    return gateValue.value;\n  };\n\n  isClientSessionEnabled = (): boolean => {\n    const isEnabled =\n      this.featureGates.filter(\n        (gate) =>\n          gate.name === FeatureGates.IsClientSessionEnabled &&\n          gate.value === true\n      ).length > 0;\n    return isEnabled;\n  };\n\n  getFeatureFlags = (): Record<string, boolean> => {\n    return Object.fromEntries(\n      Object.values(FeatureGates).map((featureGate: FeatureGates) => [\n        featureGate,\n        this.isFeatureGateEnabled(featureGate),\n      ])\n    );\n  };\n}\n\nexport interface FeatureFlagContextType {\n  featureFlagService: FeatureFlagService | undefined;\n}\n\nconst FeatureFlagServiceContext = createContext<\n  FeatureFlagContextType | undefined\n>(undefined);\n\ninterface FeatureFlagServiceProviderProps {\n  featureGates: OrgConfigFeatureGate[];\n  children: ReactNode;\n}\n\nexport const FeatureFlagServiceProvider: React.FC<\n  FeatureFlagServiceProviderProps\n> = ({ featureGates, children }) => {\n  const featureFlagService = useMemo(\n    () => new FeatureFlagService(featureGates),\n    [featureGates]\n  );\n\n  return (\n    <FeatureFlagServiceContext.Provider value={{ featureFlagService }}>\n      {children}\n    </FeatureFlagServiceContext.Provider>\n  );\n};\n\nexport const useFeatureFlagService = () => {\n  const context = useContext(FeatureFlagServiceContext);\n  if (context === undefined) {\n    throw new Error(\n      \"useFeatureFlagService must be used within a FeatureFlagServiceProvider\"\n    );\n  }\n  return context;\n};\n","import React, {\n  createContext,\n  useContext,\n  ReactNode,\n  useState,\n  useEffect,\n  useMemo,\n} from \"react\";\nimport { useAtomValue, useSetAtom } from \"jotai\";\nimport {\n  BasicOrgInfoType,\n  getOrgInfo,\n} from \"src/application/models/supportedOrgs\";\nimport { useColorsAndFrontendConfig } from \"src/hooks/useGraphQLConfig\";\nimport { orgShortNameAtom } from \"src/atoms/envive/enviveConfig\";\nimport { getAtomStore } from \"src/atoms/atomStore/atomStore\"; // Import getAtomStore\nimport { orgIdAtom } from \"src/atoms/org/graphqlConfig\"; // Import new atoms\nimport { newOrgConfigAtom } from \"src/atoms/org/newOrgConfigAtom\"; // Import newOrgConfigAtom\nimport { OrgConfigFeatureGate } from \"src/application/models/api/orgConfigResults\"; // Import OrgConfigFeatureGate\nimport { FeatureFlagServiceProvider } from \"src/contexts/featureFlagServiceContext\"; // Import FeatureFlagServiceProvider\nimport { ColorsConfigResponse, FrontendConfigResponse } from \"./graphqlContext\";\n\nexport interface NewOrgConfigContextType {\n  combinedConfig?:\n    | (BasicOrgInfoType & {\n        colorsConfig?: ColorsConfigResponse;\n        frontendConfig?: FrontendConfigResponse;\n      })\n    | null;\n  loading: boolean;\n  error: Error | null;\n}\n\nconst NewOrgConfigContext = createContext<NewOrgConfigContextType | undefined>(\n  undefined\n);\n\ninterface NewOrgConfigProviderProps {\n  children: ReactNode;\n}\n\nexport const NewOrgConfigProvider: React.FC<NewOrgConfigProviderProps> = ({\n  children,\n}) => {\n  const [oldConfig, setOldConfig] = useState<BasicOrgInfoType | undefined>();\n  const orgShortName = useAtomValue(orgShortNameAtom);\n  const setNewOrgConfig = useSetAtom(newOrgConfigAtom);\n\n  const { data: newConfig, loading, error } = useColorsAndFrontendConfig();\n\n  useEffect(() => {\n    if (orgShortName) {\n      getOrgInfo(orgShortName).then(setOldConfig);\n    }\n  }, [orgShortName]);\n\n  const combinedConfig = useMemo(() => {\n    if (!oldConfig || !newConfig) return null;\n    return { ...oldConfig, ...newConfig };\n  }, [oldConfig, newConfig]);\n\n  useEffect(() => {\n    const atomStore = getAtomStore();\n    if (combinedConfig) {\n      // TODO: Replace with actual orgId when available in combinedConfig\n      atomStore.set(orgIdAtom, \"mock-org-id\");\n\n      setNewOrgConfig(combinedConfig);\n    }\n  }, [combinedConfig, setNewOrgConfig]);\n\n  const contextValue = useMemo(() => {\n    if (!orgShortName || (loading && !oldConfig)) {\n      return { combinedConfig: null, loading: true, error: null };\n    }\n\n    if (error) {\n      return { combinedConfig: null, loading: false, error };\n    }\n\n    return { combinedConfig, loading: false, error: null };\n  }, [orgShortName, loading, error, oldConfig, combinedConfig]);\n\n  // TODO: Provide actual featureGates when available in combinedConfig\n  const featureGates: OrgConfigFeatureGate[] = [];\n\n  return (\n    <NewOrgConfigContext.Provider value={contextValue}>\n      <FeatureFlagServiceProvider featureGates={featureGates}>\n        {children}\n      </FeatureFlagServiceProvider>\n    </NewOrgConfigContext.Provider>\n  );\n};\n\nexport const useNewOrgConfigContext = () => {\n  const context = useContext(NewOrgConfigContext);\n  if (context === undefined) {\n    throw new Error(\n      \"useNewOrgConfigContext must be used within a NewOrgConfigProvider\"\n    );\n  }\n  return context;\n};\n","import { useNewOrgConfigContext } from \"src/contexts/newOrgConfigContext\";\n\nexport const useNewOrgConfig = () => {\n  const { combinedConfig, loading, error } = useNewOrgConfigContext();\n\n  return { ...combinedConfig, loading, error };\n};\n","import {\n  PLPAttributeCategory,\n  UserEventCategory,\n} from \"@spiffy-ai/commerce-api-client\";\nimport { SearchResult } from \"src/application/models/api/search\";\nimport { UserEvent, VariantInfo } from \"src/application/models\";\nimport { v4 as uuid } from \"uuid\";\n\nexport const isElementPartiallyVisible = (el?: HTMLDivElement | null) => {\n  if (!el) return false;\n\n  const rect = el.getBoundingClientRect();\n  const windowHeight =\n    window.innerHeight || document.documentElement.clientHeight;\n  const windowWidth = window.innerWidth || document.documentElement.clientWidth;\n  const verticallyVisible =\n    Math.round(rect.top) < windowHeight && Math.round(rect.bottom) > 0;\n  const horizontallyVisible =\n    Math.round(rect.left) < windowWidth && Math.round(rect.right) > 0;\n  return verticallyVisible && horizontallyVisible;\n};\n\nexport const createAppLoadedEvent = (): UserEvent => ({\n  eventId: uuid(),\n  createdAt: new Date().toISOString(),\n  category: UserEventCategory.AppLoaded,\n});\n\nexport const createVisitUserEvent = ({\n  variantInfo,\n}: {\n  variantInfo: VariantInfo;\n}): UserEvent | undefined => {\n  // this is a pdp visit event\n  if (variantInfo.variant === \"pdp\" && variantInfo.productId != null) {\n    return {\n      eventId: uuid(),\n      createdAt: new Date().toISOString(),\n      category: UserEventCategory.PdpVisit,\n      attributes: {\n        productId: variantInfo.productId,\n        parentProductId: variantInfo.parentProductId ?? \"\",\n        url: variantInfo.url ?? \"\",\n      },\n    };\n  }\n\n  // this is a plp visit event\n  if (variantInfo.variant === \"plp\" && variantInfo.plpId != null) {\n    return {\n      eventId: uuid(),\n      createdAt: new Date().toISOString(),\n      category: UserEventCategory.PlpVisit,\n      attributes: {\n        category: PLPAttributeCategory.Id,\n        attributes: {\n          id: variantInfo.plpId,\n        },\n      },\n    };\n  }\n\n  if (variantInfo.variant === \"page_visit\") {\n    return {\n      eventId: uuid(),\n      createdAt: new Date().toISOString(),\n      category: UserEventCategory.PageVisit,\n      attributes: {\n        url: variantInfo.url,\n        pageVisitCategory: variantInfo.pageVisitCategory,\n      },\n    };\n  }\n  return undefined;\n};\n\nconst parseTime = (time: string, timeZone: string) => {\n  const times = time.match(/^([0-1]?\\d):([0-5]\\d)(AM|PM)$/i);\n  const hours = times?.[1];\n  const minutes = times?.[2];\n  const period = times?.[3];\n\n  if (hours && minutes && period) {\n    const date = new Date();\n\n    // Adjust hours for AM/PM\n    let adjustedHours = 0;\n    if (period.toUpperCase() === \"PM\" && hours !== \"12\") {\n      adjustedHours = parseInt(hours) + 12;\n    }\n\n    if (period.toUpperCase() === \"AM\" && hours !== \"12\") {\n      adjustedHours = parseInt(hours);\n    }\n\n    // Create the date string with time zone\n    const formattedDate = `${date.toISOString().split(\"T\")[0]}T${String(\n      adjustedHours\n    ).padStart(2, \"0\")}:${minutes}:00`;\n    return new Date(`${formattedDate}${timeZone}`);\n  }\n};\n\nexport const isWithinBusinessHours = (\n  startTime: string,\n  endTime: string,\n  timeZone: string\n) => {\n  // Parse start and end times\n  const start = parseTime(startTime, timeZone);\n  let end = parseTime(endTime, timeZone);\n\n  if (!start || !end) {\n    return false;\n  }\n\n  let now = new Date();\n\n  // If the end date is greater than the start date, add a day to both of them\n  // This handles configurations such as startTime = 10:00PM and endTime = 7:00AM\n  if (end < start) {\n    end = new Date(end.getTime() + 24 * 60 * 60 * 1000);\n    now = new Date(now.getTime() + 24 * 60 * 60 * 1000);\n  }\n\n  // If the date conversion shifts the end date to the next day, consider comparing it across midnight\n  if (end.getUTCDate() > start.getUTCDate()) {\n    const crossingMidnight = new Date(now.getTime() + 24 * 60 * 60 * 1000);\n    return start <= now || crossingMidnight <= end;\n  }\n\n  // Check if the current time falls between start and end\n  return now >= start && now <= end;\n};\n\nexport enum SearchResultsState {\n  Loading,\n  Results,\n  NoResults,\n}\n\nexport const getSearchResultsState = (\n  isLoadingSearch: boolean,\n  searchData: SearchResult | null\n): SearchResultsState => {\n  if (isLoadingSearch) {\n    return SearchResultsState.Loading;\n  }\n  if (searchData) {\n    return SearchResultsState.Results;\n  }\n  return SearchResultsState.NoResults;\n};\n","import { useAtom, useAtomValue, useSetAtom } from \"jotai\";\nimport { useCallback, useEffect, useMemo, useRef, useState } from \"react\";\nimport { useDebounce } from \"src/hooks/useDebounce\";\nimport { useNewOrgConfig, useTrackComponentVisibleEvent } from \"src/hooks\";\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\";\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\";\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 (productList.length > 0) {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.SearchResultsViewed,\n        eventProps: {\n          queryText: searchText,\n          resultsCount: productList.length,\n        },\n      });\n    }\n  }, [productList.length, searchText, 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","import { useAtomValue } from \"jotai\";\nimport {\n  ContextEnvEnum,\n  ContextSourceEnum,\n} from \"@spiffy-ai/commerce-api-client\";\nimport { VariantInfo } from \"src/application/models\";\nimport { UserIdentityContextType } from \"src/contexts/userIdentityContext\";\nimport { variantInfoAtom } from \"src/atoms/app/variant\";\nimport {\n  contextSourceAtom,\n  envAtom as enviveEnvAtom,\n  orgShortNameAtom as enviveOrgShortNameAtom,\n} from \"src/atoms/envive/enviveConfig\";\nimport { chatIdAtom, userIdAtom } from \"src/atoms/app/index\";\nimport { useNewOrgConfigContext } from \"src/contexts/newOrgConfigContext\";\nimport { useOrgId } from \"src/hooks/useGraphQLConfig\"; // Import useOrgId\n\nexport interface AppDetails {\n  orgId: string;\n  orgShortName: string;\n  chatId: string;\n  userId: string;\n  source: ContextSourceEnum;\n  env: ContextEnvEnum;\n  variantInfo: VariantInfo;\n}\n\nexport const useAppDetails = (): AppDetails => {\n  const { orgId: fetchedOrgId } = useOrgId(); // Get orgId from useOrgId hook\n  const orgId = fetchedOrgId ?? \"\"; // Provide a default empty string if undefined\n\n  const orgShortName = useAtomValue(enviveOrgShortNameAtom) ?? \"spiffy-ai\";\n  const chatId = useAtomValue(chatIdAtom);\n  const userId = useAtomValue(userIdAtom);\n  const source = useAtomValue(contextSourceAtom) ?? ContextSourceEnum.App;\n  const env =\n    (useAtomValue(enviveEnvAtom) as ContextEnvEnum) ?? ContextEnvEnum.Dev; // Cast to ContextEnvEnum\n  const variantInfo = useAtomValue(variantInfoAtom);\n\n  return {\n    orgId,\n    orgShortName,\n    chatId,\n    userId,\n    source,\n    env,\n    variantInfo,\n  };\n};\n","import React, {\n  createContext,\n  useContext,\n  useCallback,\n  useMemo,\n  useEffect,\n} from \"react\";\nimport { useAtomValue } from \"jotai\";\nimport {\n  Configuration,\n  ResponseCategory,\n  ResponseError,\n  SearchApi,\n} from \"@spiffy-ai/commerce-api-client\";\nimport { baseUrlAtom } from \"src/atoms/envive/enviveConfig\";\nimport Logger from \"src/application/logging/logger\";\nimport { useAppDetails } from \"src/hooks/useAppDetails\";\nimport { useEnviveConfig } from \"src/contexts/enviveConfigContext\";\nimport { SessionRestartRequired } from \"src/types/exceptions/sessionExceptions\";\nimport { UnsupportedProductException } from \"src/types/exceptions/unsupportedProductExceptions\";\nimport { SearchResult, SearchParams } from \"../application/models/api/search\";\nimport {\n  setSearchServiceFunction,\n  clearSearchServiceFunction,\n} from \"../atoms/search/searchServiceAdapter\";\n\nconst transformProductResponses = (products: SearchResult[\"products\"]) =>\n  products.map((data) => ({\n    id: data.id,\n    responseId: data.response_id,\n    category: ResponseCategory.Product,\n    description: data.description,\n    imageUrl: data.image_url,\n    imageUrls: data.image_urls,\n    title: data.title,\n    url: data.url,\n    originalPrice: data.original_price,\n    salePrice: data.sale_price,\n    averageRating: data.average_rating,\n    numberReviews: data.number_reviews,\n    metadata: data.metadata,\n    isForGrid: data.is_for_grid,\n    colors: data.colors,\n    sizes: data.sizes,\n    filters: data.filters,\n  }));\n\nasync function errorResponseBody(error: ResponseError) {\n  try {\n    return await error.response.json();\n  } catch {\n    return {};\n  }\n}\n\nasync function throwSessionRestartRequiredIf(errorMsg: string, error: unknown) {\n  if (!(error instanceof ResponseError)) {\n    Logger.logInfo(errorMsg, error);\n    throw error;\n  }\n\n  const errorResponse = await errorResponseBody(error);\n  if (\n    errorResponse?.message?.toLowerCase() === \"unsupported product\" ||\n    errorResponse?.app_code?.toUpperCase() === \"PRODUCT_NOT_FOUND\"\n  ) {\n    throw new UnsupportedProductException();\n  } else if (\n    errorResponse?.app_code?.toUpperCase() === \"RESTART_SESSION\" ||\n    errorResponse?.sub_code?.toUpperCase() === \"NOT_FOUND\"\n  ) {\n    Logger.logInfo(\n      \"Session does not exist. Re-start session\",\n      error,\n      error.response,\n      errorResponse\n    );\n    throw new SessionRestartRequired();\n  }\n\n  Logger.logInfo(errorMsg, error);\n  throw error;\n}\n\ninterface SearchContextValue {\n  searchProducts: (params: SearchParams) => Promise<SearchResult>;\n  isReady: boolean;\n}\n\nconst SearchContext = createContext<SearchContextValue | undefined>(undefined);\n\nexport const SearchProvider: React.FC<{ children: React.ReactNode }> = ({\n  children,\n}) => {\n  const { orgLevelApiKey, publicKey } = useEnviveConfig();\n  const apiKey = orgLevelApiKey || publicKey; // Replicate apiKeyAtom logic\n  const appDetails = useAppDetails();\n  const baseUrl = useAtomValue(baseUrlAtom);\n\n  const isReady = Boolean(apiKey && appDetails && baseUrl);\n\n  const searchApi = useMemo(() => {\n    if (!isReady) return null;\n\n    const config: Configuration = new Configuration({\n      accessToken: apiKey,\n      basePath: baseUrl,\n      headers: {\n        \"Content-Type\": \"application/json\",\n        Accept: \"application/json\",\n      },\n    });\n    return new SearchApi(config);\n  }, [apiKey, baseUrl, isReady]);\n\n  const searchProducts = useCallback(\n    async (params: SearchParams): Promise<SearchResult> => {\n      if (!isReady || !searchApi) {\n        throw new Error(\"SearchService not ready - missing dependencies\");\n      }\n\n      try {\n        const response = await searchApi.v1SearchQueryGet({\n          query: params.query,\n          limit: params.limit,\n          org_id: appDetails.orgId,\n          user_id: appDetails.userId,\n        });\n        const {\n          products,\n          filters,\n          search_response_id: searchResponseId,\n        } = response;\n\n        return {\n          products: transformProductResponses(products) || [],\n          filters: filters || [],\n          totalProductCount: products?.length || 0,\n          searchResponseId: searchResponseId || \"\",\n        };\n      } catch (error) {\n        await throwSessionRestartRequiredIf(\"Failed to search products\", error);\n        // This part will not be reached if an exception is thrown, but it's required for type safety\n        return {\n          products: [],\n          filters: [],\n          totalProductCount: 0,\n          searchResponseId: \"\",\n        };\n      }\n    },\n    [searchApi, isReady, appDetails]\n  );\n\n  // Register the search function with the adapter for use in atoms\n  useEffect(() => {\n    if (isReady) {\n      setSearchServiceFunction(searchProducts);\n    } else {\n      clearSearchServiceFunction();\n    }\n\n    return () => {\n      clearSearchServiceFunction();\n    };\n  }, [searchProducts, isReady]);\n\n  const value = useMemo(\n    () => ({\n      searchProducts,\n      isReady,\n    }),\n    [searchProducts, isReady]\n  );\n\n  return (\n    <SearchContext.Provider value={value}>{children}</SearchContext.Provider>\n  );\n};\n\nexport const useSearchService = () => {\n  const context = useContext(SearchContext);\n  if (!context) {\n    throw new Error(\"useSearchService must be used within a SearchProvider\");\n  }\n  return context;\n};\n","import { useState, useCallback, useEffect } from \"react\";\nimport { useSearchService } from \"src/contexts/searchContext\";\nimport { SearchResult, SearchParams } from \"src/application/models/api/search\";\n\nexport const useProductSearch = () => {\n  const { searchProducts, isReady } = useSearchService();\n  const [data, setData] = useState<SearchResult | undefined>();\n  const [loading, setLoading] = useState(false);\n  const [error, setError] = useState<Error | null>(null);\n\n  const search = useCallback(\n    async (params: SearchParams) => {\n      if (!isReady) {\n        setError(new Error(\"Search service not ready - missing dependencies\"));\n        return;\n      }\n\n      setLoading(true);\n      setError(null);\n\n      try {\n        const result = await searchProducts(params);\n        setData(result);\n      } catch (err) {\n        setError(\n          err instanceof Error ? err : new Error(\"Unknown search error\")\n        );\n      } finally {\n        setLoading(false);\n      }\n    },\n    [searchProducts, isReady]\n  );\n\n  const reset = useCallback(() => {\n    setData(undefined);\n    setError(null);\n    setLoading(false);\n  }, []);\n\n  return {\n    data,\n    loading,\n    error,\n    search,\n    reset,\n    isReady,\n  };\n};\n\nexport const useSearchWithQuery = (params: SearchParams | null) => {\n  const { searchProducts, isReady } = useSearchService();\n  const [data, setData] = useState<SearchResult | undefined>();\n  const [loading, setLoading] = useState(false);\n  const [error, setError] = useState<Error | null>(null);\n\n  const executeSearch = useCallback(\n    async (searchParams: SearchParams) => {\n      if (!isReady) return;\n\n      setLoading(true);\n      setError(null);\n\n      try {\n        const result = await searchProducts(searchParams);\n        setData(result);\n      } catch (err) {\n        setError(\n          err instanceof Error ? err : new Error(\"Unknown search error\")\n        );\n      } finally {\n        setLoading(false);\n      }\n    },\n    [searchProducts, isReady]\n  );\n\n  useEffect(() => {\n    if (params && isReady) {\n      executeSearch(params);\n    }\n  }, [params, isReady, executeSearch]);\n\n  const refetch = useCallback(() => {\n    if (params) {\n      executeSearch(params);\n    }\n  }, [params, executeSearch]);\n\n  return {\n    data,\n    loading,\n    error,\n    refetch,\n    isReady,\n  };\n};\n","import React, {\n  createContext,\n  useContext,\n  useCallback,\n  useMemo,\n  useEffect,\n} from \"react\";\nimport Logger from \"../application/logging/logger\";\n\ninterface SessionStorageContextType {\n  setItem: (key: string, value: string) => void;\n  getItem: (key: string) => string | null;\n  isAvailable: boolean;\n}\n\nconst SessionStorageContext = createContext<SessionStorageContextType | null>(\n  null\n);\n\nexport const SessionStorageProvider: React.FC<{\n  children: React.ReactNode;\n}> = ({ children }) => {\n  const isAvailable = useMemo(() => {\n    try {\n      return typeof window !== \"undefined\" && !!window.sessionStorage;\n    } catch {\n      return false;\n    }\n  }, []);\n\n  useEffect(() => {\n    if (!isAvailable) {\n      Logger.logError(\"sessionStorage is not available\", undefined);\n    }\n  }, [isAvailable]);\n\n  const setItem = useCallback(\n    (key: string, value: string) => {\n      if (!isAvailable) return;\n      sessionStorage.setItem(key, value);\n      window.dispatchEvent(\n        new StorageEvent(\"storage\", { key, newValue: value })\n      );\n    },\n    [isAvailable]\n  );\n\n  const getItem = useCallback(\n    (key: string) => {\n      if (!isAvailable) return null;\n      return sessionStorage.getItem(key);\n    },\n    [isAvailable]\n  );\n\n  const value = useMemo(\n    () => ({\n      setItem,\n      getItem,\n      isAvailable,\n    }),\n    [setItem, getItem, isAvailable]\n  );\n\n  return (\n    <SessionStorageContext.Provider value={value}>\n      {children}\n    </SessionStorageContext.Provider>\n  );\n};\n\nexport const useSessionStorage = () => {\n  const context = useContext(SessionStorageContext);\n  if (!context) {\n    throw new Error(\n      \"useSessionStorage must be used within a SessionStorageProvider\"\n    );\n  }\n  return context;\n};\n","import { useState, useEffect, useCallback } from \"react\";\nimport { useSessionStorage } from \"../contexts/sessionStorageContext\";\n\nexport const useSessionStorageValue = (key: string) => {\n  const { getItem, setItem } = useSessionStorage();\n  const [value, setValue] = useState<string | null>(() => getItem(key));\n\n  useEffect(() => {\n    const handleStorageChange = (event: StorageEvent) => {\n      if (event.key === key) {\n        setValue(event.newValue);\n      }\n    };\n\n    window.addEventListener(\"storage\", handleStorageChange);\n    return () => window.removeEventListener(\"storage\", handleStorageChange);\n  }, [key, getItem]);\n\n  const updateValue = useCallback(\n    (newValue: string) => {\n      setItem(key, newValue);\n      setValue(newValue);\n    },\n    [key, setItem]\n  );\n\n  return { value, setValue: updateValue };\n};\n","import React, { createContext, useCallback, useContext, useMemo } from \"react\";\nimport { IdExtractor } from \"src/contexts/types\";\nimport { parseHref } from \"src/application/utils/urlsParser\";\n\ninterface ShopifyUrlContextType {\n  getTrimmedPathName: () => string | null;\n  getPlpOrPdpId: (extractor: IdExtractor) => string | null;\n  isOnPdpPage: () => boolean;\n  isOnPlpPage: () => boolean;\n  isReady: boolean;\n}\n\nconst ShopifyUrlContext = createContext<ShopifyUrlContextType | undefined>(\n  undefined\n);\n\nexport const ShopifyUrlProvider: React.FC<{ children: React.ReactNode }> = ({\n  children,\n}) => {\n  const isReady = true; // No external dependencies, so always ready\n\n  const getTrimmedPathName = useCallback((): string | null => {\n    let { pathname } = window.location;\n    // strip out the proxy path so local dev still works\n    pathname = pathname.replace(\"/proxy\", \"\");\n    // remove trailing hash if it exists\n    pathname = pathname.replace(/#.*$/, \"\");\n    // remove trailing /, ie. some URLs can end with '/' like /collections/cast-iron/\n    pathname = pathname.replace(/\\/$/, \"\");\n\n    if (pathname === undefined || pathname === null || pathname.length === 0) {\n      return null;\n    }\n\n    return pathname;\n  }, []);\n\n  const getPlpOrPdpId = useCallback(\n    (extractor: IdExtractor): string | null => {\n      if (extractor === \"shopify-product-variant-id\") {\n        const parsedHref = parseHref(window.location.href);\n        const variantId = parsedHref?.urlSearchParams?.get(\"variant\");\n        if (!variantId) {\n          return getPlpOrPdpId(\"shopify-product-id\");\n        }\n        return variantId;\n      }\n\n      const pathSegment =\n        extractor === \"shopify-product-id\" ? \"products\" : \"collections\";\n      const tokens = getTrimmedPathName()?.split(\"/\");\n      const idIndex = tokens?.findIndex((token) => token === pathSegment);\n      if (idIndex !== undefined && idIndex >= 0 && tokens) {\n        return decodeURIComponent(tokens[idIndex + 1]);\n      }\n      return null;\n    },\n    [getTrimmedPathName]\n  );\n\n  const isOnPdpPage = useCallback((): boolean => {\n    return getTrimmedPathName()?.includes(\"/products\") ?? false;\n  }, [getTrimmedPathName]);\n\n  const isOnPlpPage = useCallback((): boolean => {\n    return (\n      (getTrimmedPathName()?.includes(\"/collections\") &&\n        !getTrimmedPathName()?.includes(\"/products\")) ??\n      false\n    );\n  }, [getTrimmedPathName]);\n\n  const value = useMemo(\n    () => ({\n      getTrimmedPathName,\n      getPlpOrPdpId,\n      isOnPdpPage,\n      isOnPlpPage,\n      isReady,\n    }),\n    [getTrimmedPathName, getPlpOrPdpId, isOnPdpPage, isOnPlpPage, isReady]\n  );\n\n  return (\n    <ShopifyUrlContext.Provider value={value}>\n      {children}\n    </ShopifyUrlContext.Provider>\n  );\n};\n\nexport const useShopifyUrl = () => {\n  const context = useContext(ShopifyUrlContext);\n  if (!context) {\n    throw new Error(\"useShopifyUrl must be used within a ShopifyUrlProvider\");\n  }\n  return context;\n};\n","import { useCallback } from \"react\";\nimport { useShopifyUrl } from \"src/contexts/shopifyUrlContext\";\nimport { IdExtractor } from \"src/contexts/types\";\n\nexport const useShopifyUrlOperations = () => {\n  const {\n    getTrimmedPathName,\n    getPlpOrPdpId,\n    isOnPdpPage,\n    isOnPlpPage,\n    isReady,\n  } = useShopifyUrl();\n\n  return {\n    getTrimmedPathName,\n    getPlpOrPdpId,\n    isOnPdpPage,\n    isOnPlpPage,\n    isReady,\n  };\n};\n\nexport const useCurrentPageType = () => {\n  const { isOnPdpPage, isOnPlpPage, isReady } = useShopifyUrl();\n\n  const getPageType = useCallback(() => {\n    if (!isReady) return \"unknown\";\n    if (isOnPdpPage()) return \"pdp\";\n    if (isOnPlpPage()) return \"plp\";\n    return \"other\";\n  }, [isReady, isOnPdpPage, isOnPlpPage]);\n\n  return { getPageType, isReady };\n};\n\nexport const useProductId = (extractor: IdExtractor) => {\n  const { getPlpOrPdpId, isReady } = useShopifyUrl();\n\n  const productId = useCallback(() => {\n    if (!isReady) return null;\n    return getPlpOrPdpId(extractor);\n  }, [isReady, getPlpOrPdpId, extractor]);\n\n  return { productId, isReady };\n};\n","import { useMemo } from 'react';\n\nexport type Unit = 'percent' | 'pixel';\n\nexport const useSnapCalculator = (snaps: number[], maxHeight: number, unit: Unit) => {\n  const viewportHeightPx = document.documentElement.clientHeight;\n  const swipeviewHeightPx =\n    unit === 'percent' ? Math.floor(viewportHeightPx * (maxHeight / 100)) : maxHeight;\n\n  const snapsToPixels = useMemo(\n    () =>\n      snaps?.map((snap) =>\n        Math.abs(\n          (unit === 'percent' ? Math.floor(swipeviewHeightPx * (snap / 100)) : snap) -\n            swipeviewHeightPx,\n        ),\n      ),\n    [viewportHeightPx],\n  );\n\n  const getPixelToSnap = (pixels: number) => {\n    const snapIdx = snapsToPixels?.indexOf(pixels) || 0;\n    return snaps?.[snapIdx] || 0;\n  };\n\n  const getSnapToPixel = (snap: number) => {\n    const snapIdx = snaps?.indexOf(snap) || 0;\n    return snapsToPixels?.[snapIdx] || 0;\n  };\n\n  return {\n    viewportHeightPx,\n    snapsToPixels,\n    swipeviewHeightPx,\n    getPixelToSnap,\n    getSnapToPixel,\n  };\n};\n","import { useContext } from 'react';\nimport { SystemSettingsContext } from 'src/contexts';\n\nexport const useSystemSettingsContext = () => {\n  const context = useContext(SystemSettingsContext);\n\n  if (!context) {\n    throw new Error('useSystemSettingsContext must be used within a SystemSettingsContextProvider');\n  }\n\n  return { ...context };\n};\n","import { RefObject, useEffect, useRef } from \"react\";\nimport { SpiffyWidgets } from \"src/application/models/spiffyWidgets\";\nimport { useIntersection } from \"src/hooks/useIntersection\";\nimport {\n  SpiffyMetricsEventName,\n  useAmplitude,\n} from \"src/contexts/amplitudeContext\";\n\n/**\n * Tracks a component and logs an event to Amplitude when the component is visible.\n *\n * @param component - The component to track.\n * @param element - The element to track visibility of.\n * @param eventProps - Additional properties to include with the event.\n * @param eventName - The Amplitude event name to track (defaults to ChatComponentVisible).\n */\nexport const useTrackComponentVisibleEvent = (\n  component: SpiffyWidgets,\n  element: RefObject<HTMLElement>,\n  eventProps?: Record<string, unknown>,\n  eventName: SpiffyMetricsEventName = SpiffyMetricsEventName.ChatComponentVisible\n) => {\n  const isVisible = useIntersection(element, \"0px\");\n  const hasTrackedEvent = useRef(false);\n  const { trackEvent } = useAmplitude();\n\n  const componentProps = (() => {\n    if (eventName === SpiffyMetricsEventName.ChatComponentVisible) {\n      return {\n        chat_component: component,\n        ...eventProps,\n      };\n    }\n    if (eventName === SpiffyMetricsEventName.SearchComponentVisible) {\n      return {\n        search_component: component,\n        ...eventProps,\n      };\n    }\n    // Default case for other event types\n    return {\n      component: component,\n      ...eventProps,\n    };\n  })();\n\n  useEffect(() => {\n    if (isVisible && !hasTrackedEvent.current) {\n      trackEvent({\n        eventName: eventName,\n        eventProps: componentProps,\n      });\n      hasTrackedEvent.current = true;\n    }\n  }, [isVisible, component, eventProps, eventName, componentProps, trackEvent]);\n};\n","import { useAtomValue } from \"jotai\";\nimport { useEffect, useRef } from \"react\";\nimport {\n  SpiffyMetricsEventName,\n  useAmplitude,\n} from \"src/contexts/amplitudeContext\";\nimport {\n  hasParsedVariantInfoAtom,\n  variantInfoAtom,\n} from \"src/atoms/app/variant\";\n\n/**\n * Updates the default analytics properties whenever the variant info changes. This hook also\n * triggers any events that should be sent once per page visit.\n */\nexport const useUpdateAnalyticsProps = () => {\n  const variantInfo = useAtomValue(variantInfoAtom);\n  const hasInitialized = useRef(false);\n  const hasParsedVariantInfo = useAtomValue(hasParsedVariantInfoAtom);\n  const { trackEvent, setSupplementalDefaultProps } = useAmplitude();\n\n  useEffect(() => {\n    const variantInfoWithPrefix = Object.fromEntries(\n      Object.entries(variantInfo).map(([key, value]) => [\n        `variantInfo.${key}`,\n        value,\n      ])\n    );\n\n    const defaultEventProperties: Record<string, unknown> = {\n      page_variant: variantInfo.variant, // TODO: should be removed over time, after migration\n      ...variantInfoWithPrefix,\n    };\n\n    // TODO: should be removed over time, after migration\n    if (variantInfo.variant === \"pdp\") {\n      defaultEventProperties.product_id = variantInfo.productId;\n    }\n\n    // TODO: should be removed over time, after migration\n    if (variantInfo.variant === \"plp\") {\n      defaultEventProperties.plp_id = variantInfo.plpId;\n    }\n\n    // TODO: should be removed over time, after migration\n    if (variantInfo.variant === \"page_visit\") {\n      defaultEventProperties.page_visit_category =\n        variantInfo.pageVisitCategory;\n      defaultEventProperties.page_visit_url = variantInfo.url;\n    }\n\n    setSupplementalDefaultProps(defaultEventProperties);\n\n    // put any events that should be sent once per page visit here\n    if (!hasInitialized.current && hasParsedVariantInfo) {\n      trackEvent({\n        eventName: SpiffyMetricsEventName.BundleLoaded,\n      });\n      hasInitialized.current = true;\n    }\n  }, [\n    variantInfo,\n    hasParsedVariantInfo,\n    trackEvent,\n    setSupplementalDefaultProps,\n  ]);\n};\n","import { UserEventCategory } from \"@spiffy-ai/commerce-api-client\";\nimport { useAtom, useAtomValue, useSetAtom } from \"jotai\";\nimport {\n  createContext,\n  useEffect,\n  useState,\n  ReactNode,\n  useCallback,\n  useMemo,\n} from \"react\";\nimport CommerceApiClient from \"src/application/commerce-api\";\nimport { v4 as uuid } from \"uuid\";\nimport { SessionRestartRequired } from \"src/types/exceptions/sessionExceptions\";\nimport Logger from \"src/application/logging/logger\";\nimport {\n  Message,\n  MessageRole,\n  MessageType,\n  NextMessageRequest,\n  Response,\n} from \"src/application/models\";\nimport { SpiffyMetricsEventName } from \"src/contexts/amplitudeContext\";\nimport { messageFromResponse } from \"src/application/utils\";\nimport { chatIdAtom, userIdAtom, variantInfoAtom } from \"src/atoms/app\";\nimport {\n  messagesAtom,\n  requestFailureAtom,\n  responseStreamingAtom,\n  userHasRepliedAtom,\n  PerfMetricsEvents,\n  logPerfMetricAtom,\n  userEventsAtom,\n  suggestionsLoadingAtom,\n  suggestionsAtom,\n} from \"src/atoms/chat\";\nimport {\n  chatSearchProductSortingAtom,\n  chatSearchProducts,\n  chatSearchStateAtom,\n  handleSearchResultsAtom,\n  chatSearchIsLoadingAtom,\n} from \"src/atoms/search/chatSearch\";\nimport { useSystemSettingsContext } from \"src/hooks\";\nimport { createAppLoadedEvent, createVisitUserEvent } from \"src/hooks/utils\";\nimport { useMessageInterceptor } from \"src/interceptors/useMessageInterceptor\";\nimport { supportedEventAtom } from \"src/atoms/app/variant\";\nimport { chatAtom } from \"src/atoms/chat\";\nimport { getAtomStore } from \"src/atoms/atomStore\";\nimport {\n  clearUserEventAtom,\n  createResponsePayload,\n  processUserEventAtom,\n  userEventQueueAtom,\n  userQueueEventCountAtom,\n} from \"src/atoms/chat/messageQueue\";\nimport { useAmplitudeTracking } from \"src/hooks/useAmplitudeOperations\";\n\n/**\n * Record the chat assistant response in Amplitude\n *\n * @param startTimeMs The start time of the assistant response\n * @param payload The payload used to generate the response\n */\nconst recordAssistantResponse = (\n  startTimeMs: number,\n  payload: NextMessageRequest,\n  track: (\n    eventName: SpiffyMetricsEventName,\n    eventProps?: Record<string, unknown>\n  ) => void\n) => {\n  const atomStore = getAtomStore();\n  const chatState = atomStore.get(chatAtom);\n  const chatSearchState = atomStore.get(chatSearchStateAtom);\n  const searchProducts = atomStore.get(chatSearchProducts);\n  const searchProductsSort = atomStore.get(chatSearchProductSortingAtom);\n  const assistantResponseTimeMs = { start: startTimeMs, end: Date.now() };\n  let userQueryProperty: string | undefined;\n\n  if (\n    chatState.replyEventCategory === UserEventCategory.SuggestionClicked &&\n    chatState.suggestion\n  ) {\n    userQueryProperty = chatState.suggestion.content;\n  } else if (chatState.userQuery && chatState.userQuery.length > 0) {\n    userQueryProperty = chatState.userQuery;\n  }\n\n  const eventProps: Record<string, unknown> = {\n    response_time_ms:\n      assistantResponseTimeMs.end - assistantResponseTimeMs.start,\n    user_event_type: chatState.replyEventCategory,\n    user_query: userQueryProperty,\n  };\n\n  if (chatState.replyEventCategory === UserEventCategory.FormSubmitted) {\n    const lastAssistantTurn = chatState.messages\n      .filter(\n        (turn) => turn.length > 0 && turn[0].role === MessageRole.Assistant\n      )\n      .pop();\n    const formType = payload.userEvents?.find(\n      (event) => event.category === UserEventCategory.FormSubmitted\n    )?.attributes.formType;\n    const formStatus = lastAssistantTurn?.some(\n      (response) => response.type === MessageType.Order\n    );\n    eventProps.form_submitted_attributes = {\n      form_type: formType,\n      status: formStatus ? \"success\" : \"failed\",\n    };\n  }\n\n  if (chatSearchState === \"product-page\") {\n    eventProps.search_products_returned = searchProducts.length;\n    eventProps.search_products_sort_type = searchProductsSort;\n  }\n\n  track(SpiffyMetricsEventName.ChatAssistantResponse, {\n    eventProps,\n  });\n};\n\ninterface ChatContextParams {}\n\nconst ChatContext = createContext<ChatContextParams | undefined>(undefined);\n\nconst updateMessageState = (\n  message: Message,\n  lastMessage: Message,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void\n): Message => {\n  if (lastMessage == null) {\n    setMessages((prev) => [...prev, [message]]);\n    return message;\n  }\n  if (\n    lastMessage.type === MessageType.Text &&\n    message.type === MessageType.Text\n  ) {\n    const newMessage = {\n      ...lastMessage,\n      metadata: {\n        ...lastMessage.metadata,\n        content: lastMessage.metadata.content + message.metadata.content,\n      },\n    };\n    setMessages((prev) => {\n      const lastTurn = prev[prev.length - 1];\n      return [\n        ...prev.slice(0, prev.length - 1),\n        [...lastTurn.slice(0, lastTurn.length - 1), newMessage],\n      ];\n    });\n    return newMessage;\n  }\n  setMessages((prev) => [\n    ...prev.slice(0, prev.length - 1),\n    [...prev[prev.length - 1], message],\n  ]);\n  return message;\n};\n\nconst handleStreamingError = (\n  _error: unknown,\n  setRequestFailure: (failed: boolean) => void,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void\n) => {\n  setRequestFailure(true);\n  setMessages((prev) => [\n    ...prev,\n    [\n      {\n        id: uuid(),\n        role: MessageRole.Assistant,\n        type: MessageType.Text,\n        createdAt: new Date().toISOString(),\n        metadata: {\n          content:\n            \"I'm sorry! I'm having trouble right now. Please refresh the page or try again in a moment.\",\n        },\n      },\n    ],\n  ]);\n};\n\nconst processStreamingResponse = async (\n  stream: AsyncIterable<Response>,\n  messageInterceptor: {\n    intercept: (response?: Response) => boolean | undefined;\n  },\n  handleSearchResults: (message: Message) => void,\n  setMessages: (updater: (prev: Message[][]) => Message[][]) => void,\n  setSearchIsLoading: (loading: boolean) => void,\n  chatId: string\n): Promise<{ hasSearchResults: boolean }> => {\n  let lastMessage: Message | undefined;\n  let hasSearchResults = false;\n\n  for await (const response of stream) {\n    try {\n      if (messageInterceptor.intercept(response)) {\n        return { hasSearchResults };\n      }\n\n      const message = messageFromResponse(response);\n      if (!message) {\n        throw new Error(\"Failed to transform API response to client message\");\n      }\n\n      if (message.type === MessageType.ProductSearch) {\n        handleSearchResults(message);\n        hasSearchResults = true;\n        setSearchIsLoading(false); // Update search loading immediately when results are detected\n      }\n\n      lastMessage = updateMessageState(message, lastMessage!, setMessages);\n    } catch (error: unknown) {\n      Logger.logWarn(\n        `[spiffy-ai] Failed to generate responses from stream chat_id=${chatId}`,\n        error,\n        {\n          lastResponse: lastMessage,\n          response,\n        }\n      );\n    }\n  }\n\n  return { hasSearchResults };\n};\n\nconst ChatContextProvider = ({ children }: { children: ReactNode }) => {\n  const logPerfMetric = useSetAtom(logPerfMetricAtom);\n  const [widgetInitialized, setWidgetInitialized] = useState(false);\n  const setUserHasReplied = useSetAtom(userHasRepliedAtom);\n  // TODO: create atoms for setting/getting the last message turn\n  const [messages, setMessages] = useAtom<Message[][]>(messagesAtom);\n  const setUserEvents = useSetAtom(userEventsAtom);\n  const setSuggestions = useSetAtom(suggestionsAtom);\n  const [suggestionsLoading, setSuggestionsLoading] = useAtom<boolean>(\n    suggestionsLoadingAtom\n  );\n  const [responseStreaming, setResponseStreaming] = useAtom<boolean>(\n    responseStreamingAtom\n  );\n  const setRequestFailure = useSetAtom(requestFailureAtom);\n  const userEvents = useAtomValue(userEventQueueAtom);\n  const userQueueEventCount = useAtomValue(userQueueEventCountAtom);\n  const markUserEventsProcessed = useSetAtom(processUserEventAtom);\n  const clearUserEventQueue = useSetAtom(clearUserEventAtom);\n  const userId = useAtomValue(userIdAtom);\n  const chatId = useAtomValue(chatIdAtom);\n  const supportedEvent = useAtomValue(supportedEventAtom);\n  // TODO: Replace with actual orgId from useEnviveConfig or NewOrgConfigContext when available\n  const orgId = \"mock-org-id\";\n\n  const variantInfo = useAtomValue(variantInfoAtom);\n  const settingsContext = useSystemSettingsContext();\n  const messageInterceptor = useMessageInterceptor();\n  const handleSearchResults = useSetAtom(handleSearchResultsAtom);\n  const setSearchIsLoading = useSetAtom(chatSearchIsLoadingAtom);\n  const { track } = useAmplitudeTracking();\n\n  const getStreamingResponses = useCallback(\n    async (\n      payload: NextMessageRequest\n    ): Promise<{ hasSearchResults: boolean }> => {\n      logPerfMetric(PerfMetricsEvents.FirstResponseStarted);\n      const stream = CommerceApiClient.getNextResponseStreaming(payload);\n\n      try {\n        setRequestFailure(false);\n\n        const { hasSearchResults } = await processStreamingResponse(\n          stream,\n          messageInterceptor,\n          handleSearchResults,\n          setMessages,\n          setSearchIsLoading,\n          chatId\n        );\n\n        return { hasSearchResults };\n      } catch (e) {\n        handleStreamingError(e, setRequestFailure, setMessages);\n        throw e;\n      } finally {\n        logPerfMetric(PerfMetricsEvents.FirstResponseCompleted);\n      }\n    },\n    [\n      logPerfMetric,\n      setRequestFailure,\n      messageInterceptor,\n      handleSearchResults,\n      setMessages,\n      setSearchIsLoading,\n      chatId,\n    ]\n  );\n\n  const getSuggestions = useCallback(async () => {\n    logPerfMetric(PerfMetricsEvents.FirstSuggestionsStarted);\n    setSuggestionsLoading(true);\n    setSuggestions([]);\n\n    const payloadWithoutAppLoaded = createResponsePayload({\n      userEvents: [],\n      generationParams: settingsContext.generationParams,\n    });\n    const response = await CommerceApiClient.getNextSuggestions(\n      payloadWithoutAppLoaded\n    );\n\n    // sort the suggestions by shortest length so the pills can be stacked horizontally\n    setSuggestions(\n      response.sort((a, b) => a.content.length - b.content.length)\n    );\n    setSuggestionsLoading(false);\n    logPerfMetric(PerfMetricsEvents.FirstSuggestionsCompleted);\n  }, [\n    logPerfMetric,\n    setSuggestionsLoading,\n    setSuggestions,\n    settingsContext.generationParams,\n  ]);\n\n  const getResponses = useCallback(\n    async (payload?: NextMessageRequest) => {\n      try {\n        const requestPayload =\n          payload ??\n          createResponsePayload({\n            userEvents,\n            generationParams: settingsContext.generationParams,\n          });\n\n        setResponseStreaming(true);\n        setSuggestions([]);\n        const startTimeMs = Date.now();\n\n        await getStreamingResponses(requestPayload);\n\n        recordAssistantResponse(startTimeMs, requestPayload, track);\n        await getSuggestions();\n      } catch (error) {\n        Logger.logError(\"[spiffy-ai] getResponses error\", error);\n      } finally {\n        // Remove search loading management from here - it's now handled independently\n        // in the processStreamingResponse function when search results are detected\n        markUserEventsProcessed(userEvents.map(({ eventId }) => eventId));\n        setUserHasReplied(false);\n        setResponseStreaming(false);\n      }\n    },\n    [\n      userEvents,\n      settingsContext.generationParams,\n      setResponseStreaming,\n      setSuggestions,\n      getStreamingResponses,\n      markUserEventsProcessed,\n      getSuggestions,\n      setUserHasReplied,\n      track,\n    ]\n  );\n\n  useEffect(() => {\n    const processUserEvents = async () => {\n      if (responseStreaming || !widgetInitialized) {\n        return;\n      }\n\n      if (\n        (variantInfo.variant === \"pdp\" && !variantInfo.productId) ||\n        (variantInfo.variant === \"plp\" && !variantInfo.plpId) ||\n        (variantInfo.variant === \"page_visit\" && !variantInfo.url)\n      ) {\n        Logger.logDebug(\n          \"[spiffy-ai] variantInfo has invalid values, skipping...\",\n          {\n            variantInfo,\n            supportedEvent,\n          }\n        );\n        return;\n      }\n\n      Logger.logDebug(\n        `Assistants Turn is_currently_streaming=${responseStreaming} initialized=${widgetInitialized}`\n      );\n      try {\n        await getResponses();\n        Logger.logInfo(`Assistants Turn [finished]`);\n      } catch (error: unknown) {\n        Logger.logError(\"[spiffy-ai] Assistants Turn error\", error);\n      }\n    };\n    if (userQueueEventCount > 0) {\n      processUserEvents();\n    }\n  }, [\n    getResponses,\n    responseStreaming,\n    userQueueEventCount,\n    widgetInitialized,\n    variantInfo,\n    supportedEvent,\n  ]);\n\n  useEffect(() => {\n    if (widgetInitialized || responseStreaming) {\n      Logger.logDebug(\n        `[spiffy-ai] initializeWidget [skipped] is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`\n      );\n      return;\n    }\n\n    const hydrateChat = async () => {\n      try {\n        Logger.logDebug(\n          `[spiffy-ai] initializeWidget is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`\n        );\n        // on mount, try to get the responses from an active session if one exists\n        if (!orgId) {\n          throw new Error(\"orgId is not available\");\n        }\n        const { messages: existingMessages, userEvents } =\n          await CommerceApiClient.getResponses(orgId, chatId, userId);\n        setMessages([...existingMessages]);\n        setUserEvents([...userEvents]);\n        getResponses();\n      } catch (error) {\n        // no active chat session was found, start a new one\n        Logger.logInfo(\n          `Init chat [exception] chat_id=${chatId} error=${error}`,\n          error\n        );\n        if (error instanceof SessionRestartRequired) {\n          const appLoadedEvent = createAppLoadedEvent();\n          const visitEvent = createVisitUserEvent({ variantInfo });\n          setMessages([]);\n          clearUserEventQueue();\n          if (visitEvent) {\n            const payload = createResponsePayload({\n              userEvents: [appLoadedEvent, visitEvent],\n              generationParams: settingsContext.generationParams,\n            });\n            getResponses(payload);\n          }\n        }\n      } finally {\n        setWidgetInitialized(true);\n      }\n    };\n\n    hydrateChat();\n  }, []);\n\n  const onFocus = useCallback(async () => {\n    try {\n      if (!responseStreaming && !suggestionsLoading && orgId) {\n        const { messages: existingMessages } =\n          await CommerceApiClient.getResponses(orgId, chatId, userId);\n\n        if (existingMessages.length > messages.length) {\n          setMessages([...existingMessages]);\n          // TODO: Is this bug hack still necessary?\n          // If it is, come up with a better solution for it\n          // } else if (existingMessages.length === 0) {\n          //   // if there was an error during the initialization of a new session, the session would be\n          //   // created in the backend but without any messages. Retry the next_responses request with\n          //   // the current set of parameters to \"jumpstart\" the session\n          //   triggerGetResponseCall('onFocus');\n        }\n      }\n    } catch (error: unknown) {\n      Logger.logError(\"[spiffy-ai] onFocus error\", error);\n    }\n  }, [\n    responseStreaming,\n    suggestionsLoading,\n    orgId,\n    chatId,\n    userId,\n    messages.length,\n    setMessages,\n  ]);\n\n  // listen for page focus to get latest messages from the server\n  useEffect(() => {\n    window.addEventListener(\"focus\", onFocus);\n\n    return () => {\n      window.removeEventListener(\"focus\", onFocus);\n    };\n  }, [onFocus]);\n\n  const chatContext = useMemo(() => ({}), []);\n\n  return (\n    <ChatContext.Provider value={chatContext}>{children}</ChatContext.Provider>\n  );\n};\n\nexport { ChatContext, ChatContextProvider };\nexport type { ChatContextParams };\n","import React, { ReactNode } from \"react\";\nimport { useNewOrgConfig } from \"src/hooks\";\n\ninterface EnviveCssProviderProps {\n  children: ReactNode;\n}\n\nexport const EnviveCssProvider: React.FC<EnviveCssProviderProps> = ({\n  children,\n}) => {\n  const { colorsConfig, frontendConfig, loading } = useNewOrgConfig();\n  let merchantThemeCss = `* {}`;\n  if (colorsConfig && !loading) {\n    merchantThemeCss = `\n* {\n  --spiffy-colors-text-primary: ${colorsConfig.textPrimary};\n  --spiffy-colors-text-secondary: ${colorsConfig.textSecondary};\n  --spiffy-colors-text-accent: ${colorsConfig.textAccent};\n  --spiffy-colors-text-link: ${colorsConfig.textLink};\n  --spiffy-colors-text-light: ${colorsConfig.textLight};\n  --spiffy-colors-background-primary: ${colorsConfig.backgroundPrimary};\n  --spiffy-colors-background-secondary: ${colorsConfig.backgroundSecondary};\n  --spiffy-colors-background-secondary-dark: ${colorsConfig.backgroundSecondaryDark};\n  --spiffy-colors-background-tertiary: ${colorsConfig.backgroundTertiary};\n  --spiffy-colors-background-dark: ${colorsConfig.backgroundDark};\n  --spiffy-colors-background-light: ${colorsConfig.backgroundLight};\n  --spiffy-colors-background-saturated: ${colorsConfig.backgroundSaturated};\n  --spiffy-colors-border-light: ${colorsConfig.borderLight};\n  --spiffy-colors-border-medium: ${colorsConfig.borderMedium};\n  --spiffy-colors-border-dark: ${colorsConfig.borderDark};\n  --spiffy-colors-border-outline: ${colorsConfig.borderOutline};\n  --spiffy-colors-accent-primary: ${colorsConfig.accentPrimary};\n  --spiffy-colors-accent-secondary: ${colorsConfig.accentSecondary};\n}`;\n  }\n  console.log(frontendConfig);\n  return (\n    <>\n      {frontendConfig ? (\n        <style id=\"merchant-css-overrides\">{`${frontendConfig.merchantOverrideCss}`}</style>\n      ) : null}\n      {colorsConfig ? (\n        <style id=\"merchant-css-colors\">{`${merchantThemeCss}`}</style>\n      ) : null}\n\n      {children}\n    </>\n  );\n};\n","import { useAtomValue } from \"jotai\";\nimport {\n  Dispatch,\n  ReactNode,\n  SetStateAction,\n  createContext,\n  useMemo,\n  useState,\n} from \"react\";\nimport { GenerationParams } from \"src/application/models\";\nimport { baseUrlAtom } from \"src/atoms/envive/enviveConfig\";\n\n// TODO: Refactor this to a common service that handles query params (and spiffy variables all together)\nconst getChatModelName = () => {\n  const urlObj = new URL(window.location.href);\n  const params = new URLSearchParams(urlObj.search);\n  const paramsObj = Object.fromEntries(params.entries());\n  return paramsObj.llm_model_name;\n};\n\nconst defaultGenerationParams: GenerationParams = {\n  stream: true,\n  numSuggestions: 3,\n  model: getChatModelName(),\n};\n\ninterface SystemSettingsContextParams {\n  generationParams?: GenerationParams;\n  showDebugBar?: boolean;\n  endpointURL?: string;\n  setGenerationParams: Dispatch<SetStateAction<GenerationParams | undefined>>;\n}\n\ninterface SystemSettingsContextProviderProps {\n  children: ReactNode;\n  generationParams?: GenerationParams;\n  showDebugBar?: boolean;\n}\n\nconst SystemSettingsContext = createContext<\n  SystemSettingsContextParams | undefined\n>(undefined);\n\nconst SystemSettingsContextProvider = ({\n  children,\n  generationParams,\n  showDebugBar,\n}: SystemSettingsContextProviderProps) => {\n  const [params, setParams] = useState<GenerationParams | undefined>(\n    () => generationParams ?? defaultGenerationParams\n  );\n  const endpointURL = useAtomValue(baseUrlAtom);\n  const settingsContextValues = useMemo(\n    () => ({\n      generationParams: params,\n      showDebugBar,\n      setGenerationParams: setParams,\n      endpointURL,\n    }),\n    [generationParams, endpointURL, showDebugBar]\n  );\n\n  return (\n    <SystemSettingsContext.Provider value={settingsContextValues}>\n      {children}\n    </SystemSettingsContext.Provider>\n  );\n};\n\nexport {\n  SystemSettingsContextProvider,\n  SystemSettingsContext,\n  defaultGenerationParams,\n};\n"],"mappings":"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"}