@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,2980 @@
1
+ const require_chunk = require('./chunk-CUT6urMc.cjs');
2
+ const require_models = require('./models-BkXg5eIP.cjs');
3
+ const require_types = require('./types-BVSyY3Hk.cjs');
4
+ const require_graphql = require('./graphql-BNW60InJ.cjs');
5
+ const require_api = require('./api-_JaUnIUj.cjs');
6
+ const require_enviveConfig = require('./enviveConfig-BccuzS2a.cjs');
7
+ const require_atomStore = require('./atomStore-B4jIaDPd.cjs');
8
+ const require_app = require('./app-CnKRZ9RW.cjs');
9
+ const require_amplitudeContext = require('./amplitudeContext-D-7fmVh1.cjs');
10
+ const require_utils = require('./utils-D5HO61hG.cjs');
11
+ const require_graphqlConfig = require('./graphqlConfig-DicykC-o.cjs');
12
+ const require_chat = require('./chat-CO7cRkaq.cjs');
13
+ const require_org = require('./org-C2RrBVQR.cjs');
14
+ const require_search = require('./search-Df0Ifneg.cjs');
15
+ const require_globalSearch = require('./globalSearch-BFvEg53C.cjs');
16
+ const require_useMessageInterceptor = require('./useMessageInterceptor-B87e3yu3.cjs');
17
+ let react = require("react");
18
+ react = require_chunk.__toESM(react);
19
+ let jotai = require("jotai");
20
+ jotai = require_chunk.__toESM(jotai);
21
+ let __spiffy_ai_commerce_api_client = require("@spiffy-ai/commerce-api-client");
22
+ __spiffy_ai_commerce_api_client = require_chunk.__toESM(__spiffy_ai_commerce_api_client);
23
+ let react_jsx_runtime = require("react/jsx-runtime");
24
+ react_jsx_runtime = require_chunk.__toESM(react_jsx_runtime);
25
+ let uuid = require("uuid");
26
+ uuid = require_chunk.__toESM(uuid);
27
+ let ua_parser_js = require("ua-parser-js");
28
+ ua_parser_js = require_chunk.__toESM(ua_parser_js);
29
+
30
+ //#region src/contexts/cdnContext.tsx
31
+ const CdnContext = (0, react.createContext)(null);
32
+ const CdnProvider = ({ children }) => {
33
+ const cdnUrl = (0, jotai.useAtomValue)(require_enviveConfig.cdnUrlAtom) || "https://cdn.spiffy.ai/other";
34
+ const getCdnBasePath = (0, react.useCallback)(() => {
35
+ return cdnUrl;
36
+ }, [cdnUrl]);
37
+ const getAssetURL = (0, react.useCallback)((assetName, orgShortName) => {
38
+ return `${getCdnBasePath()}/assets/${orgShortName}/${assetName}`;
39
+ }, [getCdnBasePath]);
40
+ const value = (0, react.useMemo)(() => ({
41
+ cdnUrl,
42
+ getCdnBasePath,
43
+ getAssetURL
44
+ }), [
45
+ cdnUrl,
46
+ getCdnBasePath,
47
+ getAssetURL
48
+ ]);
49
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(CdnContext.Provider, {
50
+ value,
51
+ children
52
+ });
53
+ };
54
+ const useCdn = () => {
55
+ const context = (0, react.useContext)(CdnContext);
56
+ if (!context) throw new Error("useCdn must be used within a CdnProvider");
57
+ return context;
58
+ };
59
+
60
+ //#endregion
61
+ //#region src/types/exceptions/sessionExceptions.ts
62
+ var SessionRestartRequired = class extends Error {
63
+ constructor() {
64
+ super("Session restart required");
65
+ this.name = "SessionRestartRequired";
66
+ }
67
+ };
68
+
69
+ //#endregion
70
+ //#region src/types/exceptions/unsupportedProductExceptions.ts
71
+ var UnsupportedProductException = class extends Error {
72
+ constructor() {
73
+ super("Unsupported product");
74
+ this.name = "UnsupportedProduct";
75
+ }
76
+ };
77
+
78
+ //#endregion
79
+ //#region src/application/commerce-api.ts
80
+ async function errorResponseBody$1(error) {
81
+ try {
82
+ return await error.response.json();
83
+ } catch {
84
+ return {};
85
+ }
86
+ }
87
+ async function throwSessionRestartRequiredIf$1(errorMsg, error) {
88
+ if (!(error instanceof __spiffy_ai_commerce_api_client.ResponseError)) {
89
+ require_api.logger_default.logInfo(errorMsg, error);
90
+ throw error;
91
+ }
92
+ const errorResponse = await errorResponseBody$1(error);
93
+ if (errorResponse?.message?.toLowerCase() === "unsupported product" || errorResponse?.app_code?.toUpperCase() === "PRODUCT_NOT_FOUND") throw new UnsupportedProductException();
94
+ else if (errorResponse?.app_code?.toUpperCase() === "RESTART_SESSION" || errorResponse?.sub_code?.toUpperCase() === "NOT_FOUND") {
95
+ require_api.logger_default.logInfo("Session does not exist. Re-start session", error, error.response, errorResponse);
96
+ throw new SessionRestartRequired();
97
+ }
98
+ require_api.logger_default.logInfo(errorMsg, error);
99
+ throw error;
100
+ }
101
+ var CommerceApiClient = class CommerceApiClient {
102
+ static {
103
+ this.getInstance = () => {
104
+ if (!CommerceApiClient.instance) CommerceApiClient.instance = new CommerceApiClient();
105
+ return CommerceApiClient.instance;
106
+ };
107
+ }
108
+ constructor(basePath) {
109
+ this.suggestionsAbortController = new AbortController();
110
+ this.responsesAbortController = new AbortController();
111
+ const baseUrl = require_atomStore.getAtomStore().get(require_enviveConfig.baseUrlAtom);
112
+ const config = new __spiffy_ai_commerce_api_client.Configuration({
113
+ basePath: basePath || baseUrl,
114
+ headers: {
115
+ "Content-Type": "application/json",
116
+ Accept: "application/json"
117
+ }
118
+ });
119
+ this.defaultApi = new __spiffy_ai_commerce_api_client.DefaultApi(config);
120
+ this.inferenceApi = new __spiffy_ai_commerce_api_client.InferenceApi(config);
121
+ this.customerServiceApi = new __spiffy_ai_commerce_api_client.CustomerServiceApi(config);
122
+ }
123
+ static {
124
+ this.resolveUrl = async (url) => {
125
+ const atomStore = require_atomStore.getAtomStore();
126
+ const orgShortName = atomStore.get(require_enviveConfig.orgShortNameAtom);
127
+ const orgId = atomStore.get(require_graphqlConfig.orgIdAtom);
128
+ const userId = atomStore.get(require_app.userIdAtom);
129
+ const chatId = atomStore.get(require_app.chatIdAtom);
130
+ const source = atomStore.get(require_enviveConfig.contextSourceAtom);
131
+ const env = atomStore.get(require_enviveConfig.envAtom);
132
+ const featureFlagService = atomStore.get(require_graphqlConfig.featureFlagServiceAtom);
133
+ const context = {
134
+ user_id: userId ?? "",
135
+ org_id: orgId ?? "",
136
+ org_short_name: orgShortName ?? "",
137
+ chat_id: chatId ?? "",
138
+ source: source ?? __spiffy_ai_commerce_api_client.ContextSourceEnum.App,
139
+ env: env ?? __spiffy_ai_commerce_api_client.ContextEnvEnum.Dev
140
+ };
141
+ const featureGates = featureFlagService?.featureFlagService?.getFeatureFlags() || {};
142
+ const urlResolvingRequest = {
143
+ url,
144
+ context,
145
+ feature_gates: featureGates
146
+ };
147
+ return await (await CommerceApiClient.getInstance().inferenceApi.v1UrlResolvingPostRaw({ UrlResolvingRequest: urlResolvingRequest })).raw.json();
148
+ };
149
+ }
150
+ static {
151
+ this.reportSession = async (reportRequest) => {
152
+ await CommerceApiClient.getInstance().defaultApi.v1ChatsReportSessionIdPost({ ReportSessionRequest: reportRequest });
153
+ };
154
+ }
155
+ static {
156
+ this.getNextResponses = async (payload) => {
157
+ try {
158
+ return (await CommerceApiClient.getInstance().inferenceApi.v1NextResponsesPost({ NextMessageRequest: require_utils.messageRequestToCommerceMessageRequest(payload) })).map((resp) => require_models.validateResponse(resp)).map((resp) => require_utils.messageFromResponse(resp)).filter((m) => m != null);
159
+ } catch (err) {
160
+ require_api.logger_default.logInfo("Failed to get next responses", err, {
161
+ payloadContext: payload?.context,
162
+ userEvents: payload?.userEvents
163
+ });
164
+ await throwSessionRestartRequiredIf$1("Failed to get next responses", err);
165
+ return [];
166
+ }
167
+ };
168
+ }
169
+ static {
170
+ this.getNextResponseStreaming = (payload) => {
171
+ async function* generate(inferenceApi, abortController) {
172
+ try {
173
+ const response = await inferenceApi.v1NextResponsesPostRaw({ NextMessageRequest: require_utils.messageRequestToCommerceMessageRequest(payload) }, { signal: abortController.signal });
174
+ if (!response.raw.body) {
175
+ require_api.logger_default.logError("[spiffy-ai] No body in the streamed response", void 0, { response: response.raw });
176
+ return;
177
+ }
178
+ const reader = response.raw.body.getReader();
179
+ const decoder = new TextDecoder("utf-8");
180
+ let partial = "";
181
+ const safeParse = (line) => {
182
+ try {
183
+ return JSON.parse(line);
184
+ } catch (err) {
185
+ require_api.logger_default.logError("[spiffy-ai] Error parsing streamed line", err, {
186
+ line,
187
+ partial
188
+ });
189
+ partial = line;
190
+ return partial;
191
+ }
192
+ };
193
+ const processChunk = (chunk) => {
194
+ return `${partial}${chunk}`.split("\n").map((line) => line.replace(/^data: /, "").trim()).filter((line) => line !== "" && line !== "[DONE]").map(safeParse).filter((v) => v);
195
+ };
196
+ while (true) {
197
+ const { done, value } = await reader.read();
198
+ if (done) break;
199
+ const chunk = decoder.decode(value);
200
+ const parsedLines = processChunk(chunk);
201
+ for (const parsedLine of parsedLines) {
202
+ const validatedResponse = require_models.validateResponse(parsedLine);
203
+ if (validatedResponse) yield validatedResponse;
204
+ }
205
+ }
206
+ } catch (error) {
207
+ require_api.logger_default.logError("[spiffy-ai] Failed to get next streaming responses", error, {
208
+ payloadContext: payload?.context,
209
+ userEvents: payload?.userEvents
210
+ });
211
+ await throwSessionRestartRequiredIf$1("Failed to get next streaming responses", error);
212
+ }
213
+ }
214
+ CommerceApiClient.getInstance().responsesAbortController.abort();
215
+ CommerceApiClient.getInstance().responsesAbortController = new AbortController();
216
+ return generate(CommerceApiClient.getInstance().inferenceApi, CommerceApiClient.getInstance().responsesAbortController);
217
+ };
218
+ }
219
+ static {
220
+ this.getNextSuggestions = async (payload) => {
221
+ try {
222
+ CommerceApiClient.getInstance().suggestionsAbortController.abort();
223
+ CommerceApiClient.getInstance().suggestionsAbortController = new AbortController();
224
+ return (await CommerceApiClient.getInstance().inferenceApi.v1NextSuggestionsPost({ NextMessageRequest: require_utils.messageRequestToCommerceMessageRequest(payload) }, { signal: CommerceApiClient.getInstance().suggestionsAbortController.signal })).map((resp) => require_models.validateSuggestion(resp)).filter((suggestion) => suggestion != null);
225
+ } catch (error) {
226
+ require_api.logger_default.logInfo("Failed to get suggestions", error, {
227
+ payloadContext: payload?.context,
228
+ userEvents: payload?.userEvents
229
+ });
230
+ await throwSessionRestartRequiredIf$1("Failed to get suggestions", error);
231
+ return [];
232
+ }
233
+ };
234
+ }
235
+ static {
236
+ this.getResponses = async (orgId, chatId, userId) => {
237
+ let data = {
238
+ responses: [],
239
+ suggestions: [],
240
+ user_events: []
241
+ };
242
+ const request = {
243
+ org_id: orgId,
244
+ chat_id: chatId,
245
+ user_id: userId
246
+ };
247
+ try {
248
+ data = await CommerceApiClient.getInstance().defaultApi.v1GetSessionMessages(request);
249
+ } catch (error) {
250
+ await throwSessionRestartRequiredIf$1("Failed to get chat responses", error);
251
+ }
252
+ const responses = data?.responses?.map((turn) => turn.map((response) => require_models.validateResponse(response)).filter((response) => response != null));
253
+ const suggestions = data?.suggestions.map((suggestion) => require_models.validateSuggestion(suggestion)).filter((suggestion) => suggestion != null);
254
+ const userEvents = data?.user_events.map((event) => require_models.validateUserEvent(event)).filter((event) => event != null);
255
+ const formSubmittedUserEventsFormIds = userEvents.filter((event) => event.category === __spiffy_ai_commerce_api_client.UserEventCategory.FormSubmitted).map((event) => event.attributes.formResponseId);
256
+ const assistantMessages = responses.map((turn) => turn.filter((response) => !(response.category === __spiffy_ai_commerce_api_client.ResponseCategory.Form && formSubmittedUserEventsFormIds.includes(response.id))).map((response) => require_utils.messageFromResponse(response)).filter((message) => message != null)).filter((turn) => turn.length > 0);
257
+ const userMessages = userEvents.map((event) => {
258
+ if ([__spiffy_ai_commerce_api_client.UserEventCategory.QueryTyped, __spiffy_ai_commerce_api_client.UserEventCategory.Search].includes(event.category)) return [require_utils.messageFromQueryEvent(event)];
259
+ if (event.category === __spiffy_ai_commerce_api_client.UserEventCategory.SuggestionClicked) return [require_utils.messageFromSuggestionEvent(event, suggestions)];
260
+ if (event.category === __spiffy_ai_commerce_api_client.UserEventCategory.FormSubmitted) {
261
+ const formResponse = responses.flat().find((response) => response.id === event.attributes.formResponseId && event.attributes.formType !== __spiffy_ai_commerce_api_client.FormType.Escalation);
262
+ if (formResponse && formResponse.category === __spiffy_ai_commerce_api_client.ResponseCategory.Form) return [require_utils.messageFromFormSubmittedEvent(event, formResponse.attributes)];
263
+ }
264
+ return [];
265
+ }).filter((message) => message.length > 0);
266
+ const sortedMessages = [...assistantMessages, ...userMessages].sort((a, b) => new Date(a[0].createdAt).getTime() - new Date(b[0].createdAt).getTime());
267
+ return {
268
+ responses,
269
+ userEvents,
270
+ suggestions,
271
+ messages: sortedMessages
272
+ };
273
+ };
274
+ }
275
+ static {
276
+ this.isSupportedEvent = async (payload) => {
277
+ try {
278
+ const httpResponseText = await (await CommerceApiClient.getInstance().inferenceApi.v1SupportedEventPostRaw({ SupportedEventRequest: require_utils.coreSupportedEventRequestToApiRequest(payload) })).raw.text();
279
+ const httpResponseJson = JSON.parse(httpResponseText);
280
+ return {
281
+ ...httpResponseJson,
282
+ numberOfReviews: httpResponseJson.num_of_reviews,
283
+ merchant_tags: httpResponseJson.merchant_tags || []
284
+ };
285
+ } catch (err) {
286
+ require_api.logger_default.logError("Failed to get response for v1SupportedEventPost", { err });
287
+ return {
288
+ supported: false,
289
+ ready: false,
290
+ category: void 0,
291
+ collections: [],
292
+ numberOfReviews: void 0,
293
+ top_category: void 0,
294
+ merchant_tags: []
295
+ };
296
+ }
297
+ };
298
+ }
299
+ static {
300
+ this.identifyUser = async (spiffyUserId, merchantUserId, uaDetails) => {
301
+ try {
302
+ await CommerceApiClient.getInstance().defaultApi.v1AnalyticsIdentifyPost({ AnalyticsIdentifyRequest: {
303
+ user_id: spiffyUserId,
304
+ os_name: uaDetails.os,
305
+ os_version: uaDetails.osVersion,
306
+ platform: uaDetails.os,
307
+ device_id: uaDetails.deviceModel,
308
+ device_brand: uaDetails.deviceBrand,
309
+ device_manufacturer: uaDetails.deviceManufacturer,
310
+ device_model: uaDetails.deviceModel,
311
+ user_properties: {
312
+ cdp_user_id: merchantUserId,
313
+ browser: uaDetails.browser,
314
+ browser_version: uaDetails.browserVersion,
315
+ user_agent: uaDetails.userAgent
316
+ }
317
+ } });
318
+ } catch (err) {
319
+ require_api.logger_default.logError("Failed to identify user", err);
320
+ }
321
+ };
322
+ }
323
+ static {
324
+ this.mapContextSourceToV1OrgConfigGetSource = (source) => {
325
+ if (source === void 0) return void 0;
326
+ switch (source) {
327
+ case __spiffy_ai_commerce_api_client.ContextSourceEnum.Fork: return __spiffy_ai_commerce_api_client.V1OrgConfigGetSourceEnum.Fork;
328
+ case __spiffy_ai_commerce_api_client.ContextSourceEnum.Playground: return __spiffy_ai_commerce_api_client.V1OrgConfigGetSourceEnum.Playground;
329
+ case __spiffy_ai_commerce_api_client.ContextSourceEnum.App: return __spiffy_ai_commerce_api_client.V1OrgConfigGetSourceEnum.App;
330
+ case __spiffy_ai_commerce_api_client.ContextSourceEnum.Test: return __spiffy_ai_commerce_api_client.V1OrgConfigGetSourceEnum.Test;
331
+ default: return source;
332
+ }
333
+ };
334
+ }
335
+ static {
336
+ this.getOrgConfig = async (user_id) => {
337
+ try {
338
+ const atomStore = require_atomStore.getAtomStore();
339
+ const reactAppName = atomStore.get(require_enviveConfig.reactAppNameAtom);
340
+ const contextSource = atomStore.get(require_enviveConfig.contextSourceAtom);
341
+ const featureFlagService = atomStore.get(require_graphqlConfig.featureFlagServiceAtom);
342
+ const request = {
343
+ namespace: reactAppName,
344
+ user_id,
345
+ source: this.mapContextSourceToV1OrgConfigGetSource(contextSource),
346
+ include_experiments: Object.values(require_models.ProductExperiment),
347
+ include_feature_gates: Object.entries(featureFlagService?.featureFlagService?.getFeatureFlags() || {}).filter(([, isEnabled]) => isEnabled).map(([featureGateName]) => featureGateName)
348
+ };
349
+ const response = await CommerceApiClient.getInstance().defaultApi.v1OrgConfigGet(request);
350
+ return require_models.validateOrgConfigResults(response);
351
+ } catch (err) {
352
+ require_api.logger_default.logError(`Failed to get org config`, err, { err });
353
+ return;
354
+ }
355
+ };
356
+ }
357
+ static {
358
+ this.addNoteToLatestConversation = async (spiffyUserId, email, customerServiceProvider) => {
359
+ require_api.logger_default.logInfo(`addNoteToLatestConversation - user_id=${spiffyUserId} email=${email} customer_service_provider=${customerServiceProvider}`);
360
+ try {
361
+ await CommerceApiClient.getInstance().customerServiceApi.v1CustserviceAddNoteToLatestConversationPost({ AddNoteToLatestConversationRequest: {
362
+ spiffy_user_id: spiffyUserId,
363
+ email,
364
+ customer_service_provider: customerServiceProvider
365
+ } });
366
+ } catch (err) {
367
+ require_api.logger_default.logError("Failed to add note to latest conversation", { err });
368
+ }
369
+ };
370
+ }
371
+ static {
372
+ this.getCustomerServiceApi = () => CommerceApiClient.getInstance().customerServiceApi;
373
+ }
374
+ };
375
+ var commerce_api_default = CommerceApiClient;
376
+
377
+ //#endregion
378
+ //#region src/hooks/useAmplitudeOperations.ts
379
+ const useAmplitudeTracking = () => {
380
+ const { trackEvent, isReady } = require_amplitudeContext.useAmplitude();
381
+ const [loading, setLoading] = (0, react.useState)(false);
382
+ const [error, setError] = (0, react.useState)(null);
383
+ return {
384
+ track: (0, react.useCallback)(async (eventName, eventProps) => {
385
+ if (!isReady) return;
386
+ setLoading(true);
387
+ setError(null);
388
+ try {
389
+ await trackEvent({
390
+ eventName,
391
+ eventProps
392
+ });
393
+ } catch (err) {
394
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Tracking failed"));
395
+ throw err;
396
+ } finally {
397
+ setLoading(false);
398
+ }
399
+ }, [trackEvent, isReady]),
400
+ loading,
401
+ error,
402
+ isReady
403
+ };
404
+ };
405
+
406
+ //#endregion
407
+ //#region src/hooks/useBlockBackButton.ts
408
+ const useBlockBackButton = (enabled, callback) => {
409
+ (0, react.useEffect)(() => {
410
+ if (enabled && window) {
411
+ if (window.history.scrollRestoration) window.history.scrollRestoration = "manual";
412
+ window.history.pushState(null, document.title, window.location.href);
413
+ window.onpopstate = (e) => {
414
+ e.preventDefault();
415
+ window.history.pushState(null, document.title, window.location.href);
416
+ callback?.();
417
+ };
418
+ }
419
+ return () => {
420
+ if (enabled && window) {
421
+ window.history.back();
422
+ window.onpopstate = null;
423
+ window.history.scrollRestoration = "auto";
424
+ }
425
+ };
426
+ }, [enabled]);
427
+ };
428
+
429
+ //#endregion
430
+ //#region src/hooks/useCdnOperations.ts
431
+ const useCdnUrl = () => {
432
+ const { cdnUrl } = useCdn();
433
+ return cdnUrl;
434
+ };
435
+ const useCdnBasePath = () => {
436
+ const { getCdnBasePath } = useCdn();
437
+ return getCdnBasePath();
438
+ };
439
+ const useAssetUrl = (assetName, orgShortName) => {
440
+ const { getAssetURL } = useCdn();
441
+ return getAssetURL(assetName, orgShortName);
442
+ };
443
+
444
+ //#endregion
445
+ //#region src/hooks/useChatToggle.ts
446
+ const useChatToggle = () => {
447
+ const onToggle = (0, jotai.useSetAtom)(require_chat.chatOnToggleAtom);
448
+ const { isOpen } = (0, jotai.useAtomValue)(require_chat.chatAtom);
449
+ const { trackEvent } = require_amplitudeContext.useAmplitude();
450
+ const toggle = (triggerLocation, triggerId) => {
451
+ if (!isOpen) trackEvent({
452
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.ChatComponentExpanded,
453
+ eventProps: { message_metadata: {
454
+ trigger_location: triggerLocation,
455
+ trigger_id: triggerId
456
+ } }
457
+ });
458
+ else trackEvent({
459
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.ChatComponentCollapsed,
460
+ eventProps: { message_metadata: {
461
+ trigger_location: triggerLocation,
462
+ trigger_id: triggerId
463
+ } }
464
+ });
465
+ onToggle();
466
+ };
467
+ const openChat = (triggerLocation, triggerId) => {
468
+ if (!isOpen) toggle(triggerLocation, triggerId);
469
+ };
470
+ const closeChat = (triggerLocation, triggerId) => {
471
+ if (isOpen) toggle(triggerLocation, triggerId);
472
+ };
473
+ return {
474
+ toggle,
475
+ isOpen,
476
+ openChat,
477
+ closeChat
478
+ };
479
+ };
480
+
481
+ //#endregion
482
+ //#region src/hooks/useChatToggleAnalytics.ts
483
+ const useChatToggleAnalytics = () => {
484
+ const setChatOnToggle = (0, jotai.useSetAtom)(require_chat.chatOnToggleAtom);
485
+ const { track } = useAmplitudeTracking();
486
+ const toggleChat = (triggerLocation) => {
487
+ setChatOnToggle(triggerLocation, track);
488
+ };
489
+ return { toggleChat };
490
+ };
491
+
492
+ //#endregion
493
+ //#region src/hooks/useCustomerSupportHandoff.ts
494
+ /**
495
+ * Hook to call the `click` method of the merchant's customer support chat widget.
496
+ *
497
+ * @param onSwitchToAgent a function to override the function returned by the hook. This is mainly to
498
+ * preserve backward compatibility for merchants not using Kustomer and will be removed when all
499
+ * CS integrations are handled.
500
+ *
501
+ * @returns a function that searches for the customer support chat widget and calls the `click` method.
502
+ */
503
+ const useCustomerSupportHandoff = (onSwitchToAgent) => {
504
+ const onKustomerSwitch = (0, react.useCallback)(() => {
505
+ const kustomerElement = document.getElementById("kustomer-ui-sdk-iframe");
506
+ if (kustomerElement == null || !(kustomerElement instanceof HTMLIFrameElement)) {
507
+ require_api.logger_default.logError("[spiffy-ai] Kustomer iFrame element not found", void 0);
508
+ return;
509
+ }
510
+ const kustomerButton = kustomerElement.contentWindow?.document?.getElementById("rootChatIcon");
511
+ if (kustomerButton == null) {
512
+ require_api.logger_default.logError("[spiffy-ai] Kustomer button not found", void 0);
513
+ return;
514
+ }
515
+ kustomerButton.click();
516
+ }, []);
517
+ if (onSwitchToAgent != null) return { onSwitch: onSwitchToAgent };
518
+ return { onSwitch: onKustomerSwitch };
519
+ };
520
+
521
+ //#endregion
522
+ //#region src/hooks/useDebounce.ts
523
+ function useDebounce(value, delay) {
524
+ const [debouncedValue, setDebouncedValue] = (0, react.useState)(value);
525
+ (0, react.useEffect)(() => {
526
+ const handler = setTimeout(() => {
527
+ setDebouncedValue(value);
528
+ }, delay);
529
+ return () => {
530
+ clearTimeout(handler);
531
+ };
532
+ }, [value, delay]);
533
+ return debouncedValue;
534
+ }
535
+
536
+ //#endregion
537
+ //#region src/hooks/useElementObserver.ts
538
+ const useElementObserver = (selector) => {
539
+ const INITIAL_RENDER_STATE = true;
540
+ const eoRef = (0, react.useRef)(require_utils.DOMObserver.add(selector));
541
+ const [renderBlocked, setRenderBlocked] = (0, react.useState)(INITIAL_RENDER_STATE);
542
+ /**
543
+ * Fired every time the HTML element changes.
544
+ *
545
+ * @param fn
546
+ */
547
+ const onChange = (fn) => {
548
+ eoRef.current?.registerOnChange(fn);
549
+ };
550
+ /**
551
+ * Fired when the HTML element is added to the DOM.
552
+ *
553
+ * @param fn
554
+ */
555
+ const onAdd = (fn) => {
556
+ eoRef.current?.registerOnAdd(fn);
557
+ };
558
+ /**
559
+ * Fired when the HTML element is removed from the DOM.
560
+ *
561
+ * @param fn
562
+ */
563
+ const onRemove = (fn) => {
564
+ eoRef.current?.registerOnRemove(fn);
565
+ };
566
+ /**
567
+ * Fired when the class of the HTML element changes.
568
+ *
569
+ * @param fn
570
+ */
571
+ const onClassChange = (fn) => {
572
+ eoRef.current?.registerOnclassChange(fn);
573
+ };
574
+ /**
575
+ * Fired when a class is added to the HTML element.
576
+ *
577
+ * @param className
578
+ * @param fn
579
+ */
580
+ const onClassAdded = (className, fn) => {
581
+ eoRef.current?.registerOnClassAdded(className, fn);
582
+ };
583
+ /**
584
+ * Fired when a class is removed from the HTML element.
585
+ *
586
+ * @param className
587
+ * @param fn
588
+ */
589
+ const onClassRemoved = (className, fn) => {
590
+ eoRef.current?.registerOnClassRemoved(className, fn);
591
+ };
592
+ /**
593
+ * Fired when a child element is added to the HTML element.
594
+ *
595
+ * @param fn
596
+ */
597
+ const onAddChild = (fn) => {
598
+ eoRef.current?.registerOnAddChild(fn);
599
+ };
600
+ /**
601
+ * Fired when a child element is removed from the HTML element.
602
+ *
603
+ * @param fn
604
+ */
605
+ const onRemoveChild = (fn) => {
606
+ eoRef.current?.registerOnRemoveChild(fn);
607
+ };
608
+ /**
609
+ * Allows hooking event listeners to the HTML element, such as `focus`, `blur`, etc.
610
+ *
611
+ * @param event
612
+ * @param fn
613
+ */
614
+ const onEvent = (event, fn) => {
615
+ eoRef.current.registerEvent(event, fn);
616
+ };
617
+ /**
618
+ * Useful when rendering a React.js component inside the HTML element.
619
+ *
620
+ * @param fn
621
+ * @returns
622
+ */
623
+ const render = (fn) => {
624
+ if (!renderBlocked) return eoRef.current.render(fn);
625
+ };
626
+ /**
627
+ * Checks if the element exists in the DOM.
628
+ *
629
+ * @returns
630
+ */
631
+ const exists = () => !!eoRef.current.getNode();
632
+ /**
633
+ * Checks if rendering is unblocked.
634
+ *
635
+ * @returns
636
+ */
637
+ const isRendered = () => !renderBlocked;
638
+ /**
639
+ * Triggers an event for the HTML element.
640
+ *
641
+ * @param event
642
+ */
643
+ const fire = (event) => {
644
+ eoRef.current.fire(event);
645
+ };
646
+ /**
647
+ * Shows the HTML element.
648
+ *
649
+ * @returns
650
+ */
651
+ const show = () => eoRef.current.show();
652
+ /**
653
+ * Hides the HTML element.
654
+ *
655
+ * @returns
656
+ */
657
+ const hide = () => eoRef.current.hide();
658
+ /**
659
+ * Blocks the rendering of elements.
660
+ *
661
+ * @returns
662
+ */
663
+ const blockRendering = () => setRenderBlocked(true);
664
+ /**
665
+ * Unblocks the rendering of elements.
666
+ *
667
+ * @returns
668
+ */
669
+ const unblockRendering = () => setRenderBlocked(false);
670
+ /**
671
+ * Applies CSS styles to the HTML element.
672
+ *
673
+ * @param styles
674
+ */
675
+ const applyStyle = (styles) => {
676
+ const node = eoRef?.current?.getNode();
677
+ node && Object.assign(node.style, styles);
678
+ };
679
+ (0, react.useEffect)(() => {
680
+ eoRef.current.init();
681
+ eoRef.current.registerOnReset(() => setRenderBlocked(INITIAL_RENDER_STATE));
682
+ require_utils.DOMObserver.observe();
683
+ return () => require_utils.DOMObserver.remove(selector);
684
+ }, [selector.getPattern()]);
685
+ return {
686
+ targetNode: eoRef.current.getNode(),
687
+ onChange,
688
+ onAdd,
689
+ onRemove,
690
+ onClassChange,
691
+ onClassAdded,
692
+ onClassRemoved,
693
+ onAddChild,
694
+ onRemoveChild,
695
+ onEvent,
696
+ blockRendering,
697
+ unblockRendering,
698
+ exists,
699
+ isRendered,
700
+ render,
701
+ fire,
702
+ show,
703
+ hide,
704
+ applyStyle
705
+ };
706
+ };
707
+
708
+ //#endregion
709
+ //#region src/hooks/useGrabAndScroll.ts
710
+ const animateHorizontalScroll = ({ element, duration, targetScroll, multiply = 1, direction, callback, offset = 0 }) => {
711
+ const start = element.scrollLeft;
712
+ const distance = (targetScroll - start) * multiply;
713
+ const startTime = performance.now();
714
+ function easeOutSine(x) {
715
+ return Math.sin(x * Math.PI / 2);
716
+ }
717
+ function scrollStep(currentTime) {
718
+ const timeElapsed = currentTime - startTime;
719
+ const progress = Math.min(timeElapsed / duration, 1);
720
+ const easing = easeOutSine(progress);
721
+ const step = start + distance * easing;
722
+ const canScroll = (direction === "rt" ? element.scrollLeft < step : element.scrollLeft > step) || !direction;
723
+ if (step > 0 && canScroll) element.scrollTo(step, 0);
724
+ if (timeElapsed < duration) requestAnimationFrame(scrollStep);
725
+ else if (element.scrollLeft + offset === element.scrollWidth) callback?.("rt");
726
+ else if (element.scrollLeft <= 1) callback?.("lt");
727
+ else callback?.("ct");
728
+ }
729
+ requestAnimationFrame(scrollStep);
730
+ };
731
+ const useGrabAndScroll = (enabled, chunkWidth, speed = 400, offset = 0) => {
732
+ const containerRef = (0, react.useRef)(null);
733
+ const [leftArrow, setLeftArrow] = (0, react.useState)(false);
734
+ const [rightArrow, setRightArrow] = (0, react.useState)(true);
735
+ const handleArrows = (position) => {
736
+ switch (position) {
737
+ case "lt":
738
+ setLeftArrow(false);
739
+ setRightArrow(true);
740
+ break;
741
+ case "rt":
742
+ setLeftArrow(true);
743
+ setRightArrow(false);
744
+ break;
745
+ default:
746
+ setLeftArrow(true);
747
+ setRightArrow(true);
748
+ }
749
+ };
750
+ const animationTrigger = () => {
751
+ if (enabled && containerRef.current) {
752
+ const dist = (containerRef?.current?.scrollLeft || 0) / chunkWidth;
753
+ const targetScroll = chunkWidth * (Math.floor(dist) + (dist % 1 > .5 ? 1 : 0));
754
+ animateHorizontalScroll({
755
+ element: containerRef.current,
756
+ targetScroll,
757
+ duration: speed,
758
+ offset,
759
+ callback: handleArrows
760
+ });
761
+ }
762
+ };
763
+ const onNext = (cardsToSlide) => {
764
+ if (containerRef.current) {
765
+ const targetScroll = containerRef.current.scrollLeft + chunkWidth;
766
+ animateHorizontalScroll({
767
+ element: containerRef.current,
768
+ targetScroll,
769
+ duration: speed,
770
+ direction: "rt",
771
+ multiply: cardsToSlide,
772
+ offset,
773
+ callback: handleArrows
774
+ });
775
+ }
776
+ };
777
+ const onPrevious = (cardsToSlide) => {
778
+ if (containerRef.current) {
779
+ const targetScroll = containerRef.current.scrollLeft - chunkWidth;
780
+ animateHorizontalScroll({
781
+ element: containerRef.current,
782
+ targetScroll,
783
+ duration: speed,
784
+ direction: "lt",
785
+ multiply: cardsToSlide,
786
+ offset,
787
+ callback: handleArrows
788
+ });
789
+ }
790
+ };
791
+ return {
792
+ containerRef,
793
+ leftArrow,
794
+ rightArrow,
795
+ animationTrigger,
796
+ onNext,
797
+ onPrevious
798
+ };
799
+ };
800
+
801
+ //#endregion
802
+ //#region src/contexts/graphqlContext.tsx
803
+ const GraphQLContext = (0, react.createContext)(null);
804
+ const colorsAndFrontendConfigQuery = () => `
805
+ query ($version: String = "${require_graphql.configVersion()}") {
806
+ me {
807
+ getProductsConfigByVersion(version: $version) {
808
+ frontend { values }
809
+ colors { values }
810
+ }
811
+ }
812
+ }
813
+ `;
814
+ const GraphQLProvider = ({ children }) => {
815
+ const apiKey = (0, jotai.useAtomValue)(require_enviveConfig.orgLevelApiKeyAtom);
816
+ const baseUrl = (0, jotai.useAtomValue)(require_enviveConfig.baseUrlAtom);
817
+ const isReady = Boolean(apiKey && baseUrl);
818
+ const executeQuery = (0, react.useCallback)(async (query, variables) => {
819
+ if (!isReady) throw new Error("GraphQL client not ready - missing apiKey or baseUrl");
820
+ const response = await fetch(`${baseUrl}/v1/graphql`, {
821
+ method: "POST",
822
+ headers: {
823
+ "Content-Type": "application/json",
824
+ Authorization: `Bearer ${apiKey}`
825
+ },
826
+ body: JSON.stringify({
827
+ query,
828
+ variables
829
+ })
830
+ });
831
+ if (!response.ok) throw new Error(`GraphQL request failed: ${response.statusText}`);
832
+ const result = await response.json();
833
+ if (result.errors) throw new Error(`GraphQL errors: ${JSON.stringify(result.errors)}`);
834
+ return result.data;
835
+ }, [
836
+ apiKey,
837
+ baseUrl,
838
+ isReady
839
+ ]);
840
+ const getOrgId = (0, react.useCallback)(async () => {
841
+ const response = await executeQuery(require_models.getMerchantOrgIdQuery);
842
+ return require_models.validateGraphQLOrgId(response.me.org?.id);
843
+ }, [executeQuery]);
844
+ const getColorsAndFrontendConfig = (0, react.useCallback)(async () => {
845
+ try {
846
+ const query = await colorsAndFrontendConfigQuery();
847
+ if (!query) throw new Error("Colors and frontend config query is not defined");
848
+ const response = await executeQuery(query);
849
+ const colorsConfig = response.me.getProductsConfigByVersion?.colors?.values;
850
+ const frontendConfig = response.me.getProductsConfigByVersion?.frontend?.values;
851
+ const transformedColorConfig = require_models.transformSnakeToCamel(colorsConfig);
852
+ const transformedFrontendConfig = require_models.transformSnakeToCamel(frontendConfig);
853
+ return {
854
+ colorsConfig: transformedColorConfig,
855
+ frontendConfig: transformedFrontendConfig
856
+ };
857
+ } catch (err) {
858
+ require_api.logger_default.logError("Error fetching graphql colors and frontend config", err);
859
+ return {
860
+ colorsConfig: void 0,
861
+ frontendConfig: void 0
862
+ };
863
+ }
864
+ }, [executeQuery]);
865
+ const value = (0, react.useMemo)(() => ({
866
+ executeQuery,
867
+ getOrgId,
868
+ getColorsAndFrontendConfig,
869
+ isReady
870
+ }), [
871
+ executeQuery,
872
+ getOrgId,
873
+ getColorsAndFrontendConfig,
874
+ isReady
875
+ ]);
876
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(GraphQLContext.Provider, {
877
+ value,
878
+ children
879
+ });
880
+ };
881
+ const useGraphQLClient = () => {
882
+ const context = (0, react.useContext)(GraphQLContext);
883
+ if (!context) throw new Error("useGraphQLClient must be used within a GraphQLProvider");
884
+ return context;
885
+ };
886
+
887
+ //#endregion
888
+ //#region src/hooks/useGraphQLConfig.ts
889
+ const useColorsAndFrontendConfig = () => {
890
+ const { getColorsAndFrontendConfig, isReady } = useGraphQLClient();
891
+ const [data, setData] = (0, react.useState)({});
892
+ const [loading, setLoading] = (0, react.useState)(false);
893
+ const [error, setError] = (0, react.useState)(null);
894
+ const fetchConfig = (0, react.useCallback)(async () => {
895
+ if (!isReady) return;
896
+ setLoading(true);
897
+ setError(null);
898
+ try {
899
+ const result = await getColorsAndFrontendConfig();
900
+ setData(result);
901
+ } catch (err) {
902
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown error"));
903
+ } finally {
904
+ setLoading(false);
905
+ }
906
+ }, [getColorsAndFrontendConfig, isReady]);
907
+ (0, react.useEffect)(() => {
908
+ fetchConfig();
909
+ }, [fetchConfig]);
910
+ return {
911
+ data,
912
+ loading,
913
+ error,
914
+ refetch: fetchConfig
915
+ };
916
+ };
917
+ const useOrgId = () => {
918
+ const { getOrgId, isReady } = useGraphQLClient();
919
+ const [orgId, setOrgId] = (0, react.useState)();
920
+ const [loading, setLoading] = (0, react.useState)(false);
921
+ const [error, setError] = (0, react.useState)(null);
922
+ (0, react.useEffect)(() => {
923
+ if (!isReady) return;
924
+ const fetchOrgId = async () => {
925
+ setLoading(true);
926
+ setError(null);
927
+ try {
928
+ const id = await getOrgId();
929
+ setOrgId(id);
930
+ } catch (err) {
931
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown error"));
932
+ } finally {
933
+ setLoading(false);
934
+ }
935
+ };
936
+ fetchOrgId();
937
+ }, [getOrgId, isReady]);
938
+ return {
939
+ orgId,
940
+ loading,
941
+ error
942
+ };
943
+ };
944
+
945
+ //#endregion
946
+ //#region src/contexts/userIdentityContext.tsx
947
+ const getUserAgentDetails = () => {
948
+ const result = new ua_parser_js.default().getResult();
949
+ return {
950
+ os: result?.os?.name,
951
+ osVersion: result?.os?.version,
952
+ deviceBrand: result?.device?.vendor,
953
+ deviceManufacturer: result?.device?.vendor,
954
+ deviceModel: result?.device?.model,
955
+ browser: result?.browser?.name,
956
+ browserVersion: result?.browser?.version,
957
+ userAgent: result?.ua
958
+ };
959
+ };
960
+ const UserIdentityContext = (0, react.createContext)(void 0);
961
+ const UserIdentityProvider = ({ children }) => {
962
+ const { getItem, setItem, isAvailable: localStorageIsReady } = require_enviveConfig.useLocalStorage();
963
+ const [isReady, setIsReady] = (0, react.useState)(false);
964
+ (0, react.useEffect)(() => {
965
+ setIsReady(localStorageIsReady);
966
+ }, [localStorageIsReady]);
967
+ const USER_ID_OVERRIDE_KEY = "v1-spiffy-user-id-override";
968
+ const USER_ID_DEFAULT_KEY = "v1-spiffy-user-id-default";
969
+ const getUserIdOverrideFromLocalStorage = (0, react.useCallback)(() => {
970
+ return getItem(USER_ID_OVERRIDE_KEY) ?? void 0;
971
+ }, [getItem]);
972
+ const getUserIdDefaultFromLocalStorage = (0, react.useCallback)(() => {
973
+ return getItem(USER_ID_DEFAULT_KEY) ?? void 0;
974
+ }, [getItem]);
975
+ const setUserIdDefaultInLocalStorage = (0, react.useCallback)((userId) => {
976
+ require_api.logger_default.logInfo(`setUserIdDefaultInLocalStorage - Setting user_id=${userId}`);
977
+ setItem(USER_ID_DEFAULT_KEY, userId);
978
+ return userId;
979
+ }, [setItem, USER_ID_DEFAULT_KEY]);
980
+ const setUserIdOverrideInLocalStorage = (0, react.useCallback)((userId) => {
981
+ require_api.logger_default.logInfo(`setUserIdOverrideInLocalStorage - Setting user_id=${userId}`);
982
+ setItem(USER_ID_OVERRIDE_KEY, userId);
983
+ return userId;
984
+ }, [setItem, USER_ID_OVERRIDE_KEY]);
985
+ const clearUserIdOverrideInLocalStorage = (0, react.useCallback)(() => {
986
+ require_api.logger_default.logInfo(`clearUserIdOverrideInLocalStorage - Clearing user_id`);
987
+ setItem(USER_ID_OVERRIDE_KEY, "");
988
+ }, [setItem, USER_ID_OVERRIDE_KEY]);
989
+ const getUserIdOrDefault = (0, react.useCallback)(() => {
990
+ const userIdOverride = getUserIdOverrideFromLocalStorage();
991
+ if (userIdOverride) return userIdOverride;
992
+ const defaultUserId = getUserIdDefaultFromLocalStorage();
993
+ if (defaultUserId) return defaultUserId;
994
+ return setUserIdDefaultInLocalStorage(`spiffy-user-id-${(0, uuid.v4)()}`);
995
+ }, [
996
+ getUserIdOverrideFromLocalStorage,
997
+ getUserIdDefaultFromLocalStorage,
998
+ setUserIdDefaultInLocalStorage
999
+ ]);
1000
+ const identifyUser = (0, react.useCallback)(async () => {
1001
+ if (!isReady) {
1002
+ require_api.logger_default.logWarn("[UserIdentityContext] Context not ready, skipping identifyUser", void 0);
1003
+ return;
1004
+ }
1005
+ try {
1006
+ const cdpUserId = "UNKNOWN_CDP_USER_ID";
1007
+ const userId = getUserIdOrDefault();
1008
+ const userAgentDetails = getUserAgentDetails();
1009
+ await commerce_api_default.identifyUser(userId, cdpUserId, userAgentDetails);
1010
+ } catch (error) {
1011
+ require_api.logger_default.logError("[spiffy-ai] Error identifying user", error);
1012
+ }
1013
+ }, [isReady, getUserIdOrDefault]);
1014
+ const value = (0, react.useMemo)(() => ({
1015
+ identifyUser,
1016
+ getUserIdOrDefault,
1017
+ getUserIdOverrideFromLocalStorage,
1018
+ getUserIdDefaultFromLocalStorage,
1019
+ setUserIdDefaultInLocalStorage,
1020
+ setUserIdOverrideInLocalStorage,
1021
+ clearUserIdOverrideInLocalStorage,
1022
+ isReady
1023
+ }), [
1024
+ identifyUser,
1025
+ getUserIdOrDefault,
1026
+ getUserIdOverrideFromLocalStorage,
1027
+ getUserIdDefaultFromLocalStorage,
1028
+ setUserIdDefaultInLocalStorage,
1029
+ setUserIdOverrideInLocalStorage,
1030
+ clearUserIdOverrideInLocalStorage,
1031
+ isReady
1032
+ ]);
1033
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(UserIdentityContext.Provider, {
1034
+ value,
1035
+ children
1036
+ });
1037
+ };
1038
+ const useUserIdentity = () => {
1039
+ const context = (0, react.useContext)(UserIdentityContext);
1040
+ if (!context) throw new Error("useUserIdentity must be used within a UserIdentityProvider");
1041
+ return context;
1042
+ };
1043
+
1044
+ //#endregion
1045
+ //#region src/hooks/useIdentifyUser.ts
1046
+ const useIdentifyUser = () => {
1047
+ const { identifyUser, isReady } = useUserIdentity();
1048
+ const [loading, setLoading] = (0, react.useState)(false);
1049
+ const [error, setError] = (0, react.useState)(null);
1050
+ const executeIdentifyUser = (0, react.useCallback)(async () => {
1051
+ if (!isReady) {
1052
+ setError(/* @__PURE__ */ new Error("UserIdentityContext not ready."));
1053
+ return;
1054
+ }
1055
+ setLoading(true);
1056
+ setError(null);
1057
+ try {
1058
+ await identifyUser();
1059
+ } catch (err) {
1060
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown error during user identification."));
1061
+ throw err;
1062
+ } finally {
1063
+ setLoading(false);
1064
+ }
1065
+ }, [identifyUser, isReady]);
1066
+ return {
1067
+ loading,
1068
+ error,
1069
+ executeIdentifyUser,
1070
+ isReady
1071
+ };
1072
+ };
1073
+
1074
+ //#endregion
1075
+ //#region src/hooks/useImageResolver.ts
1076
+ var ImageResolver = class {};
1077
+ var MerchantImageResolver = class {
1078
+ static {
1079
+ this.imageResolverMap = /* @__PURE__ */ new Map();
1080
+ }
1081
+ static loadMapping() {
1082
+ if (this.imageResolverMap.size === 0) {
1083
+ this.imageResolverMap.set(require_types.OrgShortName.Spanx, new ShopifyImageResolver());
1084
+ this.imageResolverMap.set(require_types.OrgShortName.SpanxStaging, new ShopifyImageResolver());
1085
+ this.imageResolverMap.set(require_types.OrgShortName.UniqueVintage, new ShopifyImageResolver());
1086
+ }
1087
+ return this.imageResolverMap;
1088
+ }
1089
+ static get(name) {
1090
+ return this.loadMapping().get(name);
1091
+ }
1092
+ };
1093
+ var ShopifyImageResolver = class extends ImageResolver {
1094
+ resolve(url, size) {
1095
+ const pattern = /_\d+x\.jpg/;
1096
+ const urlHasPrefix = pattern.test(url);
1097
+ const newSizePrefix = `_${size}x.jpg`;
1098
+ if (urlHasPrefix) return url.replace(pattern, newSizePrefix);
1099
+ return url.replace(".jpg", newSizePrefix);
1100
+ }
1101
+ };
1102
+ const useImageResolver = () => {
1103
+ const orgShortName = (0, jotai.useAtomValue)(require_enviveConfig.orgShortNameAtom);
1104
+ const resolve = (image, size) => {
1105
+ if (image && size && orgShortName) return MerchantImageResolver.get(orgShortName)?.resolve(image, size) || image;
1106
+ return image;
1107
+ };
1108
+ return { resolve };
1109
+ };
1110
+
1111
+ //#endregion
1112
+ //#region src/hooks/useIntersection.ts
1113
+ const useIntersection = (element, rootMargin) => {
1114
+ const [isVisible, setIsVisible] = (0, react.useState)(false);
1115
+ (0, react.useEffect)(() => {
1116
+ const current = element?.current;
1117
+ const observer = new IntersectionObserver(([entry]) => {
1118
+ setIsVisible(entry.isIntersecting);
1119
+ }, { rootMargin });
1120
+ if (current) observer?.observe(current);
1121
+ return () => {
1122
+ if (current) observer.unobserve(current);
1123
+ };
1124
+ }, []);
1125
+ return isVisible;
1126
+ };
1127
+
1128
+ //#endregion
1129
+ //#region src/hooks/useIsSmallScreen.ts
1130
+ const useIsSmallScreen = () => {
1131
+ const [isSmall, setIsSmall] = (0, react.useState)(false);
1132
+ (0, react.useEffect)(() => {
1133
+ const mediaQuery = window.matchMedia("(max-width: 479px)");
1134
+ setIsSmall(mediaQuery.matches);
1135
+ const handleResize = (event) => {
1136
+ setIsSmall(event.matches);
1137
+ };
1138
+ mediaQuery.addEventListener("change", handleResize);
1139
+ return () => mediaQuery.removeEventListener("change", handleResize);
1140
+ }, []);
1141
+ return isSmall;
1142
+ };
1143
+
1144
+ //#endregion
1145
+ //#region src/hooks/useLocalStorageOperations.ts
1146
+ const useLocalStorageValue = (key) => {
1147
+ const { getItem, setItem, attachListener, detachListener } = require_enviveConfig.useLocalStorage();
1148
+ const [value, setValue] = (0, react.useState)(() => getItem(key));
1149
+ (0, react.useEffect)(() => {
1150
+ const listener = {
1151
+ storageKey: key,
1152
+ listener: (event) => {
1153
+ setValue(event.newValue);
1154
+ }
1155
+ };
1156
+ attachListener(listener);
1157
+ return () => detachListener(listener);
1158
+ }, [
1159
+ key,
1160
+ attachListener,
1161
+ detachListener
1162
+ ]);
1163
+ const updateValue = (0, react.useCallback)((newValue) => {
1164
+ setItem(key, newValue);
1165
+ setValue(newValue);
1166
+ }, [key, setItem]);
1167
+ return {
1168
+ value,
1169
+ setValue: updateValue
1170
+ };
1171
+ };
1172
+ const useSpiffyFeatureFlag = () => {
1173
+ const { setSpiffyOnFeatureFlag } = require_enviveConfig.useLocalStorage();
1174
+ const { value } = useLocalStorageValue(require_enviveConfig.LocalStorageKeys.SpiffyOnOverride);
1175
+ const setFlag = (0, react.useCallback)((flag) => {
1176
+ setSpiffyOnFeatureFlag(flag);
1177
+ }, [setSpiffyOnFeatureFlag]);
1178
+ return {
1179
+ value: value === "true" ? true : value === "false" ? false : null,
1180
+ setFlag
1181
+ };
1182
+ };
1183
+ const useEnviveFeatureFlag = () => {
1184
+ const { setItem, getItem } = require_enviveConfig.useLocalStorage();
1185
+ const { value } = useLocalStorageValue(require_enviveConfig.LocalStorageKeys.EnviveOnOverride);
1186
+ const setFlag = (0, react.useCallback)((flag) => {
1187
+ if (flag === true) setItem(require_enviveConfig.LocalStorageKeys.EnviveOnOverride, "true");
1188
+ else if (flag === false) setItem(require_enviveConfig.LocalStorageKeys.EnviveOnOverride, "false");
1189
+ }, [setItem]);
1190
+ return {
1191
+ value: value === "true" ? true : value === "false" ? false : null,
1192
+ setFlag
1193
+ };
1194
+ };
1195
+ const useLocalStorageListener = (key, callback) => {
1196
+ const { attachListener, detachListener } = require_enviveConfig.useLocalStorage();
1197
+ (0, react.useEffect)(() => {
1198
+ const listener = {
1199
+ storageKey: key,
1200
+ listener: callback
1201
+ };
1202
+ attachListener(listener);
1203
+ return () => detachListener(listener);
1204
+ }, [
1205
+ key,
1206
+ callback,
1207
+ attachListener,
1208
+ detachListener
1209
+ ]);
1210
+ };
1211
+
1212
+ //#endregion
1213
+ //#region src/hooks/useMessageFilter.ts
1214
+ const useMessageFilter = () => {
1215
+ const findMessageIndex = ({ msgs, type, role }) => {
1216
+ let lastIndex = -1;
1217
+ msgs.forEach((subArray, index) => {
1218
+ subArray.forEach((obj) => {
1219
+ if (obj.type === type || obj.role === role) lastIndex = index;
1220
+ });
1221
+ });
1222
+ return lastIndex;
1223
+ };
1224
+ const removePreviousDiscussions = (msgs, index) => {
1225
+ if (index > -1) {
1226
+ const lastMessages = msgs.slice(index);
1227
+ return lastMessages.length > 0 ? lastMessages : msgs;
1228
+ }
1229
+ return msgs;
1230
+ };
1231
+ const getFilteredMessages = (msgs, skipFilter) => {
1232
+ const messageMap = msgs.reduce((acc, msg) => {
1233
+ acc[msg[0].id] = msg;
1234
+ return acc;
1235
+ }, {});
1236
+ const deduplicatedMsgs = Object.values(messageMap);
1237
+ if (!skipFilter) {
1238
+ const idx = findMessageIndex({
1239
+ msgs: deduplicatedMsgs,
1240
+ type: require_models.MessageType.Separator
1241
+ });
1242
+ return removePreviousDiscussions(deduplicatedMsgs, idx);
1243
+ }
1244
+ return deduplicatedMsgs;
1245
+ };
1246
+ return {
1247
+ findMessageIndex,
1248
+ removePreviousDiscussions,
1249
+ getFilteredMessages
1250
+ };
1251
+ };
1252
+
1253
+ //#endregion
1254
+ //#region src/hooks/useMessageScrollObserver.ts
1255
+ const useMessageScrollObserver = (boxRef, scrollRef, onScrollChange) => {
1256
+ const calculateScrollHeight = () => {
1257
+ const boxHeight = boxRef?.current?.getBoundingClientRect().height || 0;
1258
+ const scrollHeight = scrollRef?.current?.getBoundingClientRect().height || 0;
1259
+ return boxHeight - scrollHeight;
1260
+ };
1261
+ const updateState = () => {
1262
+ const scrollHeight = calculateScrollHeight();
1263
+ if (scrollHeight > 0) onScrollChange(scrollHeight);
1264
+ };
1265
+ (0, react.useEffect)(() => {
1266
+ let boxRO = null;
1267
+ let scrollRO = null;
1268
+ if (scrollRef?.current) {
1269
+ boxRO = new ResizeObserver(updateState);
1270
+ boxRO.observe(scrollRef?.current);
1271
+ }
1272
+ if (boxRef?.current) {
1273
+ scrollRO = new ResizeObserver(updateState);
1274
+ scrollRO.observe(boxRef?.current);
1275
+ }
1276
+ return () => {
1277
+ if (scrollRef?.current && boxRO) boxRO.unobserve(scrollRef?.current);
1278
+ if (scrollRO && boxRef?.current) scrollRO?.unobserve(boxRef?.current);
1279
+ };
1280
+ }, []);
1281
+ };
1282
+
1283
+ //#endregion
1284
+ //#region src/contexts/featureFlagServiceContext.tsx
1285
+ var FeatureFlagService = class {
1286
+ constructor(featureGates) {
1287
+ this.isFeatureGateEnabled = (featureGate) => {
1288
+ const gateValue = this.featureGates.find((gate) => gate.name === featureGate);
1289
+ if (gateValue == null || gateValue.value == null) {
1290
+ require_api.logger_default.logDebug(`[spiffy-ai] isFeatureGateEnabled featureGate:${featureGate} value is undefined - returning false`);
1291
+ return false;
1292
+ }
1293
+ return gateValue.value;
1294
+ };
1295
+ this.isClientSessionEnabled = () => {
1296
+ return this.featureGates.filter((gate) => gate.name === require_models.FeatureGates.IsClientSessionEnabled && gate.value === true).length > 0;
1297
+ };
1298
+ this.getFeatureFlags = () => {
1299
+ return Object.fromEntries(Object.values(require_models.FeatureGates).map((featureGate) => [featureGate, this.isFeatureGateEnabled(featureGate)]));
1300
+ };
1301
+ this.featureGates = featureGates;
1302
+ }
1303
+ };
1304
+ const FeatureFlagServiceContext = (0, react.createContext)(void 0);
1305
+ const FeatureFlagServiceProvider = ({ featureGates, children }) => {
1306
+ const featureFlagService = (0, react.useMemo)(() => new FeatureFlagService(featureGates), [featureGates]);
1307
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(FeatureFlagServiceContext.Provider, {
1308
+ value: { featureFlagService },
1309
+ children
1310
+ });
1311
+ };
1312
+ const useFeatureFlagService = () => {
1313
+ const context = (0, react.useContext)(FeatureFlagServiceContext);
1314
+ if (context === void 0) throw new Error("useFeatureFlagService must be used within a FeatureFlagServiceProvider");
1315
+ return context;
1316
+ };
1317
+
1318
+ //#endregion
1319
+ //#region src/contexts/newOrgConfigContext.tsx
1320
+ const NewOrgConfigContext = (0, react.createContext)(void 0);
1321
+ const NewOrgConfigProvider = ({ children }) => {
1322
+ const [oldConfig, setOldConfig] = (0, react.useState)();
1323
+ const orgShortName = (0, jotai.useAtomValue)(require_enviveConfig.orgShortNameAtom);
1324
+ const setNewOrgConfig = (0, jotai.useSetAtom)(require_org.newOrgConfigAtom);
1325
+ const { data: newConfig, loading, error } = useColorsAndFrontendConfig();
1326
+ (0, react.useEffect)(() => {
1327
+ if (orgShortName) require_types.getOrgInfo(orgShortName).then(setOldConfig);
1328
+ }, [orgShortName]);
1329
+ const combinedConfig = (0, react.useMemo)(() => {
1330
+ if (!oldConfig || !newConfig) return null;
1331
+ return {
1332
+ ...oldConfig,
1333
+ ...newConfig
1334
+ };
1335
+ }, [oldConfig, newConfig]);
1336
+ (0, react.useEffect)(() => {
1337
+ const atomStore = require_atomStore.getAtomStore();
1338
+ if (combinedConfig) {
1339
+ atomStore.set(require_graphqlConfig.orgIdAtom, "mock-org-id");
1340
+ setNewOrgConfig(combinedConfig);
1341
+ }
1342
+ }, [combinedConfig, setNewOrgConfig]);
1343
+ const contextValue = (0, react.useMemo)(() => {
1344
+ if (!orgShortName || loading && !oldConfig) return {
1345
+ combinedConfig: null,
1346
+ loading: true,
1347
+ error: null
1348
+ };
1349
+ if (error) return {
1350
+ combinedConfig: null,
1351
+ loading: false,
1352
+ error
1353
+ };
1354
+ return {
1355
+ combinedConfig,
1356
+ loading: false,
1357
+ error: null
1358
+ };
1359
+ }, [
1360
+ orgShortName,
1361
+ loading,
1362
+ error,
1363
+ oldConfig,
1364
+ combinedConfig
1365
+ ]);
1366
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(NewOrgConfigContext.Provider, {
1367
+ value: contextValue,
1368
+ children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(FeatureFlagServiceProvider, {
1369
+ featureGates: [],
1370
+ children
1371
+ })
1372
+ });
1373
+ };
1374
+ const useNewOrgConfigContext = () => {
1375
+ const context = (0, react.useContext)(NewOrgConfigContext);
1376
+ if (context === void 0) throw new Error("useNewOrgConfigContext must be used within a NewOrgConfigProvider");
1377
+ return context;
1378
+ };
1379
+
1380
+ //#endregion
1381
+ //#region src/hooks/useNewOrgConfig.ts
1382
+ const useNewOrgConfig = () => {
1383
+ const { combinedConfig, loading, error } = useNewOrgConfigContext();
1384
+ return {
1385
+ ...combinedConfig,
1386
+ loading,
1387
+ error
1388
+ };
1389
+ };
1390
+
1391
+ //#endregion
1392
+ //#region src/hooks/utils.ts
1393
+ const isElementPartiallyVisible = (el) => {
1394
+ if (!el) return false;
1395
+ const rect = el.getBoundingClientRect();
1396
+ const windowHeight = window.innerHeight || document.documentElement.clientHeight;
1397
+ const windowWidth = window.innerWidth || document.documentElement.clientWidth;
1398
+ const verticallyVisible = Math.round(rect.top) < windowHeight && Math.round(rect.bottom) > 0;
1399
+ const horizontallyVisible = Math.round(rect.left) < windowWidth && Math.round(rect.right) > 0;
1400
+ return verticallyVisible && horizontallyVisible;
1401
+ };
1402
+ const createAppLoadedEvent = () => ({
1403
+ eventId: (0, uuid.v4)(),
1404
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1405
+ category: __spiffy_ai_commerce_api_client.UserEventCategory.AppLoaded
1406
+ });
1407
+ const createVisitUserEvent = ({ variantInfo }) => {
1408
+ if (variantInfo.variant === "pdp" && variantInfo.productId != null) return {
1409
+ eventId: (0, uuid.v4)(),
1410
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1411
+ category: __spiffy_ai_commerce_api_client.UserEventCategory.PdpVisit,
1412
+ attributes: {
1413
+ productId: variantInfo.productId,
1414
+ parentProductId: variantInfo.parentProductId ?? "",
1415
+ url: variantInfo.url ?? ""
1416
+ }
1417
+ };
1418
+ if (variantInfo.variant === "plp" && variantInfo.plpId != null) return {
1419
+ eventId: (0, uuid.v4)(),
1420
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1421
+ category: __spiffy_ai_commerce_api_client.UserEventCategory.PlpVisit,
1422
+ attributes: {
1423
+ category: __spiffy_ai_commerce_api_client.PLPAttributeCategory.Id,
1424
+ attributes: { id: variantInfo.plpId }
1425
+ }
1426
+ };
1427
+ if (variantInfo.variant === "page_visit") return {
1428
+ eventId: (0, uuid.v4)(),
1429
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
1430
+ category: __spiffy_ai_commerce_api_client.UserEventCategory.PageVisit,
1431
+ attributes: {
1432
+ url: variantInfo.url,
1433
+ pageVisitCategory: variantInfo.pageVisitCategory
1434
+ }
1435
+ };
1436
+ };
1437
+ const parseTime = (time, timeZone) => {
1438
+ const times = time.match(/^([0-1]?\d):([0-5]\d)(AM|PM)$/i);
1439
+ const hours = times?.[1];
1440
+ const minutes = times?.[2];
1441
+ const period = times?.[3];
1442
+ if (hours && minutes && period) {
1443
+ const date = /* @__PURE__ */ new Date();
1444
+ let adjustedHours = 0;
1445
+ if (period.toUpperCase() === "PM" && hours !== "12") adjustedHours = parseInt(hours) + 12;
1446
+ if (period.toUpperCase() === "AM" && hours !== "12") adjustedHours = parseInt(hours);
1447
+ const formattedDate = `${date.toISOString().split("T")[0]}T${String(adjustedHours).padStart(2, "0")}:${minutes}:00`;
1448
+ return /* @__PURE__ */ new Date(`${formattedDate}${timeZone}`);
1449
+ }
1450
+ };
1451
+ const isWithinBusinessHours = (startTime, endTime, timeZone) => {
1452
+ const start = parseTime(startTime, timeZone);
1453
+ let end = parseTime(endTime, timeZone);
1454
+ if (!start || !end) return false;
1455
+ let now = /* @__PURE__ */ new Date();
1456
+ if (end < start) {
1457
+ end = new Date(end.getTime() + 1440 * 60 * 1e3);
1458
+ now = new Date(now.getTime() + 1440 * 60 * 1e3);
1459
+ }
1460
+ if (end.getUTCDate() > start.getUTCDate()) {
1461
+ const crossingMidnight = new Date(now.getTime() + 1440 * 60 * 1e3);
1462
+ return start <= now || crossingMidnight <= end;
1463
+ }
1464
+ return now >= start && now <= end;
1465
+ };
1466
+ let SearchResultsState = /* @__PURE__ */ function(SearchResultsState$1) {
1467
+ SearchResultsState$1[SearchResultsState$1["Loading"] = 0] = "Loading";
1468
+ SearchResultsState$1[SearchResultsState$1["Results"] = 1] = "Results";
1469
+ SearchResultsState$1[SearchResultsState$1["NoResults"] = 2] = "NoResults";
1470
+ return SearchResultsState$1;
1471
+ }({});
1472
+ const getSearchResultsState = (isLoadingSearch, searchData) => {
1473
+ if (isLoadingSearch) return SearchResultsState.Loading;
1474
+ if (searchData) return SearchResultsState.Results;
1475
+ return SearchResultsState.NoResults;
1476
+ };
1477
+
1478
+ //#endregion
1479
+ //#region src/hooks/useSearch.tsx
1480
+ const useSearch = () => {
1481
+ const config = useNewOrgConfig();
1482
+ const orgShortName = (0, jotai.useAtomValue)(require_enviveConfig.orgShortNameAtom);
1483
+ const { data: searchData, loading: isLoadingSearch } = (0, jotai.useAtomValue)(require_search.searchAtom);
1484
+ const productList = (0, jotai.useAtomValue)(require_search.filteredSearchProductsAtom);
1485
+ const performSearch = (0, jotai.useSetAtom)(require_search.performSearchAtom);
1486
+ const [{ results: autocompleteResults, isLoading: isLoadingAutocomplete }, setAutocompleteState] = (0, jotai.useAtom)(require_globalSearch.autocompleteStateAtom);
1487
+ const [{ query }] = (0, jotai.useAtom)(require_search.searchParamsAtom);
1488
+ const [isFilterOpen, setIsFilterOpen] = (0, jotai.useAtom)(require_globalSearch.isFilterOpenAtom);
1489
+ const [selectedFilterOptions] = (0, jotai.useAtom)(require_search.searchSelectedFiltersAtom);
1490
+ const addFilter = (0, jotai.useSetAtom)(require_search.addSearchFilterAtom);
1491
+ const removeFilter = (0, jotai.useSetAtom)(require_search.removeSearchFilterAtom);
1492
+ const [productSorting, setProductSorting] = (0, jotai.useAtom)(require_search.searchProductSortingAtom);
1493
+ const clearFilters = (0, jotai.useSetAtom)(require_search.clearSearchFiltersAtom);
1494
+ const searchFilters = (0, jotai.useAtomValue)(require_search.searchFiltersAtom);
1495
+ const [isDirty, setIsDirty] = (0, react.useState)(true);
1496
+ const [focusedIndex, setFocusedIndex] = (0, react.useState)(-1);
1497
+ const [focusedOptionId, setFocusedOptionId] = (0, react.useState)(void 0);
1498
+ const [searchText, setSearchText] = (0, react.useState)(query || "");
1499
+ const searchResultsRef = (0, react.useRef)(null);
1500
+ const debouncedSearchText = useDebounce(searchText, 200);
1501
+ const searchResultsState = getSearchResultsState(isLoadingSearch, searchData);
1502
+ const dynamicFilters = searchData?.filters || [];
1503
+ const safeProductCardConfig = config?.frontendConfig?.uiConfigs?.productCardConfig || {
1504
+ variant: "minimal",
1505
+ hoverVariant: "none",
1506
+ layoutVariant: "square"
1507
+ };
1508
+ const safeMerchantShortName = orgShortName || "";
1509
+ const availableDynamicFilters = (0, react.useMemo)(() => {
1510
+ return dynamicFilters.filter((dynamicFilterName) => !selectedFilterOptions.some((option) => option.id === `dynamic:${dynamicFilterName}`)).map((dynamicFilterName) => ({
1511
+ name: dynamicFilterName,
1512
+ displayName: require_search.formatFilterDisplayName(dynamicFilterName)
1513
+ }));
1514
+ }, [dynamicFilters, selectedFilterOptions]);
1515
+ const filters = (0, react.useMemo)(() => {
1516
+ return [{
1517
+ filterId: "sort",
1518
+ displayName: "SORT",
1519
+ items: [
1520
+ {
1521
+ filterItemId: String(require_search.ProductSorting.FEATURED),
1522
+ displayName: "Relevance",
1523
+ productCount: 0,
1524
+ isSelected: productSorting === require_search.ProductSorting.FEATURED
1525
+ },
1526
+ {
1527
+ filterItemId: String(require_search.ProductSorting.PRICE_ASC),
1528
+ displayName: "Price: Low to High",
1529
+ productCount: 0,
1530
+ isSelected: productSorting === require_search.ProductSorting.PRICE_ASC
1531
+ },
1532
+ {
1533
+ filterItemId: String(require_search.ProductSorting.PRICE_DESC),
1534
+ displayName: "Price: High to Low",
1535
+ productCount: 0,
1536
+ isSelected: productSorting === require_search.ProductSorting.PRICE_DESC
1537
+ }
1538
+ ]
1539
+ }, ...searchFilters];
1540
+ }, [productSorting, searchFilters]);
1541
+ const filterButtonText = (0, react.useMemo)(() => {
1542
+ const selectedCount = filters.reduce((acc, filter) => {
1543
+ if (filter.filterId === "sort") return acc;
1544
+ return acc + filter.items.filter((item) => item.isSelected).length;
1545
+ }, 0);
1546
+ if (selectedCount === 0) return "Filter & Sort";
1547
+ return `Filter & Sort (${selectedCount})`;
1548
+ }, [filters]);
1549
+ const { trackEvent } = require_amplitudeContext.useAmplitude();
1550
+ const handleToggleDynamicFilter = (0, react.useCallback)(({ filter, dynamicFilterDisplayName }) => {
1551
+ trackEvent({
1552
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchFilterClicked,
1553
+ eventProps: {
1554
+ filterType: "Dynamic",
1555
+ filterValue: filter,
1556
+ queryText: searchText
1557
+ }
1558
+ });
1559
+ addFilter(require_search.createFilterOption("dynamic", filter, dynamicFilterDisplayName));
1560
+ }, [
1561
+ addFilter,
1562
+ searchText,
1563
+ trackEvent
1564
+ ]);
1565
+ const handleRemoveFilter = (0, react.useCallback)((filter) => {
1566
+ removeFilter(filter.id);
1567
+ }, [removeFilter]);
1568
+ const handleSubmitSearch = (0, react.useCallback)(async () => {
1569
+ if (searchText.trim()) {
1570
+ trackEvent({
1571
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchQuerySubmitted,
1572
+ eventProps: {
1573
+ searchOrigin: require_models.SpiffyWidgets.SearchResults,
1574
+ queryText: searchText.trim()
1575
+ },
1576
+ alsoSendToGoogleAnalytics: true
1577
+ });
1578
+ const url = new URL(window.location.href);
1579
+ url.searchParams.set("esq", searchText.trim());
1580
+ window.history.pushState({}, "", url);
1581
+ performSearch({ query: searchText.trim() });
1582
+ }
1583
+ }, [
1584
+ performSearch,
1585
+ searchText,
1586
+ trackEvent
1587
+ ]);
1588
+ const handleAutocompleteSelect = (0, react.useCallback)((suggestion) => {
1589
+ setSearchText(suggestion);
1590
+ handleSubmitSearch();
1591
+ }, [handleSubmitSearch, setSearchText]);
1592
+ const handleKeyDown = (0, react.useCallback)((event) => {
1593
+ if (event.key === "ArrowDown") {
1594
+ event.preventDefault();
1595
+ const newIndex = (focusedIndex + 1) % autocompleteResults.length;
1596
+ setFocusedIndex(newIndex);
1597
+ setFocusedOptionId(`option-${newIndex}`);
1598
+ } else if (event.key === "ArrowUp") {
1599
+ event.preventDefault();
1600
+ const newIndex = (focusedIndex - 1 + autocompleteResults.length) % autocompleteResults.length;
1601
+ setFocusedIndex(newIndex);
1602
+ setFocusedOptionId(`option-${newIndex}`);
1603
+ } else if (event.key === "Enter") if (focusedIndex === -1) {
1604
+ event.preventDefault();
1605
+ handleSubmitSearch();
1606
+ } else {
1607
+ event.preventDefault();
1608
+ const suggestionText = autocompleteResults[focusedIndex];
1609
+ handleAutocompleteSelect(suggestionText);
1610
+ }
1611
+ else if (event.key === "Escape") {
1612
+ event.preventDefault();
1613
+ setFocusedIndex(-1);
1614
+ setFocusedOptionId(void 0);
1615
+ }
1616
+ }, [
1617
+ autocompleteResults,
1618
+ focusedIndex,
1619
+ handleAutocompleteSelect,
1620
+ handleSubmitSearch
1621
+ ]);
1622
+ const handleSearchInputChange = (newValue) => {
1623
+ if (newValue.length === 1) trackEvent({
1624
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchInputStarted,
1625
+ eventProps: { searchOrigin: require_models.SpiffyWidgets.SearchResults }
1626
+ });
1627
+ setSearchText(newValue);
1628
+ setIsDirty(true);
1629
+ };
1630
+ const handleSelectFilterItem = (0, react.useCallback)(({ filterId, filterItemId, isSelected, displayName }) => {
1631
+ if (filterId === "sort") {
1632
+ const newSort = filterItemId;
1633
+ trackEvent({
1634
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchSortClicked,
1635
+ eventProps: {
1636
+ sortType: newSort,
1637
+ queryText: searchText
1638
+ }
1639
+ });
1640
+ setProductSorting(newSort);
1641
+ } else if (!isSelected) removeFilter(`${filterId}:${filterItemId}`);
1642
+ else {
1643
+ trackEvent({
1644
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchFilterClicked,
1645
+ eventProps: {
1646
+ filterType: "Static",
1647
+ filterCategory: filterId,
1648
+ filterValue: filterItemId,
1649
+ queryText: searchText
1650
+ }
1651
+ });
1652
+ addFilter(require_search.createFilterOption(filterId, filterItemId, displayName));
1653
+ }
1654
+ }, [
1655
+ addFilter,
1656
+ removeFilter,
1657
+ setProductSorting,
1658
+ searchText,
1659
+ trackEvent
1660
+ ]);
1661
+ const handleClearAllFilters = (0, react.useCallback)(() => {
1662
+ setProductSorting(require_search.ProductSorting.FEATURED);
1663
+ clearFilters();
1664
+ }, [setProductSorting, clearFilters]);
1665
+ useTrackComponentVisibleEvent(require_models.SpiffyWidgets.SearchResults, searchResultsRef, {}, require_amplitudeContext.SpiffyMetricsEventName.SearchComponentVisible);
1666
+ (0, react.useEffect)(() => {
1667
+ if (productList.length > 0) trackEvent({
1668
+ eventName: require_amplitudeContext.SpiffyMetricsEventName.SearchResultsViewed,
1669
+ eventProps: {
1670
+ queryText: searchText,
1671
+ resultsCount: productList.length
1672
+ }
1673
+ });
1674
+ }, [
1675
+ productList.length,
1676
+ searchText,
1677
+ trackEvent
1678
+ ]);
1679
+ (0, react.useEffect)(() => {
1680
+ if (query && query !== searchText) setSearchText(query);
1681
+ }, [query]);
1682
+ (0, react.useEffect)(() => {
1683
+ const esq = new URLSearchParams(window.location.search).get("esq");
1684
+ if (esq) {
1685
+ setSearchText(esq);
1686
+ performSearch({ query: esq });
1687
+ }
1688
+ }, [performSearch]);
1689
+ const fetchAutocompleteSuggestions = (_query) => {
1690
+ return Promise.resolve([]);
1691
+ };
1692
+ (0, react.useEffect)(() => {
1693
+ if (fetchAutocompleteSuggestions === void 0) return;
1694
+ if (!isDirty || debouncedSearchText.length <= 2) {
1695
+ setAutocompleteState({
1696
+ results: [],
1697
+ isLoading: false
1698
+ });
1699
+ return;
1700
+ }
1701
+ setAutocompleteState((prev) => ({
1702
+ ...prev,
1703
+ isLoading: true
1704
+ }));
1705
+ const fetchData = async () => {
1706
+ try {
1707
+ const results = await fetchAutocompleteSuggestions?.(debouncedSearchText);
1708
+ setAutocompleteState({
1709
+ results: results ?? [],
1710
+ isLoading: false
1711
+ });
1712
+ } catch (error) {
1713
+ require_api.logger_default.logError("Failed to fetch autocomplete suggestions:", error);
1714
+ setAutocompleteState({
1715
+ results: [],
1716
+ isLoading: false
1717
+ });
1718
+ }
1719
+ };
1720
+ fetchData();
1721
+ }, [
1722
+ debouncedSearchText,
1723
+ isDirty,
1724
+ setAutocompleteState
1725
+ ]);
1726
+ return {
1727
+ searchData,
1728
+ searchResponseId: searchData?.searchResponseId ?? "",
1729
+ merchantShortName: safeMerchantShortName,
1730
+ productCardConfig: safeProductCardConfig,
1731
+ productList,
1732
+ autocompleteResults,
1733
+ searchFilters: filters,
1734
+ availableDynamicFilters,
1735
+ selectedFilterOptions,
1736
+ searchText,
1737
+ searchResultsState,
1738
+ isLoadingAutocomplete,
1739
+ isLoadingSearch,
1740
+ isFilterOpen,
1741
+ isDirty,
1742
+ focusedIndex,
1743
+ focusedOptionId,
1744
+ filterButtonText,
1745
+ onSearchInputChange: handleSearchInputChange,
1746
+ onSubmitSearch: handleSubmitSearch,
1747
+ onAutocompleteSelect: handleAutocompleteSelect,
1748
+ onKeyDown: handleKeyDown,
1749
+ onToggleDynamicFilter: handleToggleDynamicFilter,
1750
+ onSelectFilterItem: handleSelectFilterItem,
1751
+ onRemoveFilter: handleRemoveFilter,
1752
+ onClearAllFilters: handleClearAllFilters,
1753
+ setIsFilterOpen,
1754
+ searchResultsRef
1755
+ };
1756
+ };
1757
+
1758
+ //#endregion
1759
+ //#region src/hooks/useAppDetails.ts
1760
+ const useAppDetails = () => {
1761
+ const { orgId: fetchedOrgId } = useOrgId();
1762
+ const orgId = fetchedOrgId ?? "";
1763
+ const orgShortName = (0, jotai.useAtomValue)(require_enviveConfig.orgShortNameAtom) ?? "spiffy-ai";
1764
+ const chatId = (0, jotai.useAtomValue)(require_app.chatIdAtom);
1765
+ const userId = (0, jotai.useAtomValue)(require_app.userIdAtom);
1766
+ const source = (0, jotai.useAtomValue)(require_enviveConfig.contextSourceAtom) ?? __spiffy_ai_commerce_api_client.ContextSourceEnum.App;
1767
+ const env = (0, jotai.useAtomValue)(require_enviveConfig.envAtom) ?? __spiffy_ai_commerce_api_client.ContextEnvEnum.Dev;
1768
+ const variantInfo = (0, jotai.useAtomValue)(require_app.variantInfoAtom);
1769
+ return {
1770
+ orgId,
1771
+ orgShortName,
1772
+ chatId,
1773
+ userId,
1774
+ source,
1775
+ env,
1776
+ variantInfo
1777
+ };
1778
+ };
1779
+
1780
+ //#endregion
1781
+ //#region src/contexts/searchContext.tsx
1782
+ const transformProductResponses = (products) => products.map((data) => ({
1783
+ id: data.id,
1784
+ responseId: data.response_id,
1785
+ category: __spiffy_ai_commerce_api_client.ResponseCategory.Product,
1786
+ description: data.description,
1787
+ imageUrl: data.image_url,
1788
+ imageUrls: data.image_urls,
1789
+ title: data.title,
1790
+ url: data.url,
1791
+ originalPrice: data.original_price,
1792
+ salePrice: data.sale_price,
1793
+ averageRating: data.average_rating,
1794
+ numberReviews: data.number_reviews,
1795
+ metadata: data.metadata,
1796
+ isForGrid: data.is_for_grid,
1797
+ colors: data.colors,
1798
+ sizes: data.sizes,
1799
+ filters: data.filters
1800
+ }));
1801
+ async function errorResponseBody(error) {
1802
+ try {
1803
+ return await error.response.json();
1804
+ } catch {
1805
+ return {};
1806
+ }
1807
+ }
1808
+ async function throwSessionRestartRequiredIf(errorMsg, error) {
1809
+ if (!(error instanceof __spiffy_ai_commerce_api_client.ResponseError)) {
1810
+ require_api.logger_default.logInfo(errorMsg, error);
1811
+ throw error;
1812
+ }
1813
+ const errorResponse = await errorResponseBody(error);
1814
+ if (errorResponse?.message?.toLowerCase() === "unsupported product" || errorResponse?.app_code?.toUpperCase() === "PRODUCT_NOT_FOUND") throw new UnsupportedProductException();
1815
+ else if (errorResponse?.app_code?.toUpperCase() === "RESTART_SESSION" || errorResponse?.sub_code?.toUpperCase() === "NOT_FOUND") {
1816
+ require_api.logger_default.logInfo("Session does not exist. Re-start session", error, error.response, errorResponse);
1817
+ throw new SessionRestartRequired();
1818
+ }
1819
+ require_api.logger_default.logInfo(errorMsg, error);
1820
+ throw error;
1821
+ }
1822
+ const SearchContext = (0, react.createContext)(void 0);
1823
+ const SearchProvider = ({ children }) => {
1824
+ const { orgLevelApiKey, publicKey } = require_amplitudeContext.useEnviveConfig();
1825
+ const apiKey = orgLevelApiKey || publicKey;
1826
+ const appDetails = useAppDetails();
1827
+ const baseUrl = (0, jotai.useAtomValue)(require_enviveConfig.baseUrlAtom);
1828
+ const isReady = Boolean(apiKey && appDetails && baseUrl);
1829
+ const searchApi = (0, react.useMemo)(() => {
1830
+ if (!isReady) return null;
1831
+ const config = new __spiffy_ai_commerce_api_client.Configuration({
1832
+ accessToken: apiKey,
1833
+ basePath: baseUrl,
1834
+ headers: {
1835
+ "Content-Type": "application/json",
1836
+ Accept: "application/json"
1837
+ }
1838
+ });
1839
+ return new __spiffy_ai_commerce_api_client.SearchApi(config);
1840
+ }, [
1841
+ apiKey,
1842
+ baseUrl,
1843
+ isReady
1844
+ ]);
1845
+ const searchProducts = (0, react.useCallback)(async (params) => {
1846
+ if (!isReady || !searchApi) throw new Error("SearchService not ready - missing dependencies");
1847
+ try {
1848
+ const { products, filters, search_response_id: searchResponseId } = await searchApi.v1SearchQueryGet({
1849
+ query: params.query,
1850
+ limit: params.limit,
1851
+ org_id: appDetails.orgId,
1852
+ user_id: appDetails.userId
1853
+ });
1854
+ return {
1855
+ products: transformProductResponses(products) || [],
1856
+ filters: filters || [],
1857
+ totalProductCount: products?.length || 0,
1858
+ searchResponseId: searchResponseId || ""
1859
+ };
1860
+ } catch (error) {
1861
+ await throwSessionRestartRequiredIf("Failed to search products", error);
1862
+ return {
1863
+ products: [],
1864
+ filters: [],
1865
+ totalProductCount: 0,
1866
+ searchResponseId: ""
1867
+ };
1868
+ }
1869
+ }, [
1870
+ searchApi,
1871
+ isReady,
1872
+ appDetails
1873
+ ]);
1874
+ (0, react.useEffect)(() => {
1875
+ if (isReady) require_search.setSearchServiceFunction(searchProducts);
1876
+ else require_search.clearSearchServiceFunction();
1877
+ return () => {
1878
+ require_search.clearSearchServiceFunction();
1879
+ };
1880
+ }, [searchProducts, isReady]);
1881
+ const value = (0, react.useMemo)(() => ({
1882
+ searchProducts,
1883
+ isReady
1884
+ }), [searchProducts, isReady]);
1885
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(SearchContext.Provider, {
1886
+ value,
1887
+ children
1888
+ });
1889
+ };
1890
+ const useSearchService = () => {
1891
+ const context = (0, react.useContext)(SearchContext);
1892
+ if (!context) throw new Error("useSearchService must be used within a SearchProvider");
1893
+ return context;
1894
+ };
1895
+
1896
+ //#endregion
1897
+ //#region src/hooks/useSearchOperations.ts
1898
+ const useProductSearch = () => {
1899
+ const { searchProducts, isReady } = useSearchService();
1900
+ const [data, setData] = (0, react.useState)();
1901
+ const [loading, setLoading] = (0, react.useState)(false);
1902
+ const [error, setError] = (0, react.useState)(null);
1903
+ const search = (0, react.useCallback)(async (params) => {
1904
+ if (!isReady) {
1905
+ setError(/* @__PURE__ */ new Error("Search service not ready - missing dependencies"));
1906
+ return;
1907
+ }
1908
+ setLoading(true);
1909
+ setError(null);
1910
+ try {
1911
+ const result = await searchProducts(params);
1912
+ setData(result);
1913
+ } catch (err) {
1914
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown search error"));
1915
+ } finally {
1916
+ setLoading(false);
1917
+ }
1918
+ }, [searchProducts, isReady]);
1919
+ const reset = (0, react.useCallback)(() => {
1920
+ setData(void 0);
1921
+ setError(null);
1922
+ setLoading(false);
1923
+ }, []);
1924
+ return {
1925
+ data,
1926
+ loading,
1927
+ error,
1928
+ search,
1929
+ reset,
1930
+ isReady
1931
+ };
1932
+ };
1933
+ const useSearchWithQuery = (params) => {
1934
+ const { searchProducts, isReady } = useSearchService();
1935
+ const [data, setData] = (0, react.useState)();
1936
+ const [loading, setLoading] = (0, react.useState)(false);
1937
+ const [error, setError] = (0, react.useState)(null);
1938
+ const executeSearch = (0, react.useCallback)(async (searchParams) => {
1939
+ if (!isReady) return;
1940
+ setLoading(true);
1941
+ setError(null);
1942
+ try {
1943
+ const result = await searchProducts(searchParams);
1944
+ setData(result);
1945
+ } catch (err) {
1946
+ setError(err instanceof Error ? err : /* @__PURE__ */ new Error("Unknown search error"));
1947
+ } finally {
1948
+ setLoading(false);
1949
+ }
1950
+ }, [searchProducts, isReady]);
1951
+ (0, react.useEffect)(() => {
1952
+ if (params && isReady) executeSearch(params);
1953
+ }, [
1954
+ params,
1955
+ isReady,
1956
+ executeSearch
1957
+ ]);
1958
+ const refetch = (0, react.useCallback)(() => {
1959
+ if (params) executeSearch(params);
1960
+ }, [params, executeSearch]);
1961
+ return {
1962
+ data,
1963
+ loading,
1964
+ error,
1965
+ refetch,
1966
+ isReady
1967
+ };
1968
+ };
1969
+
1970
+ //#endregion
1971
+ //#region src/contexts/sessionStorageContext.tsx
1972
+ const SessionStorageContext = (0, react.createContext)(null);
1973
+ const SessionStorageProvider = ({ children }) => {
1974
+ const isAvailable = (0, react.useMemo)(() => {
1975
+ try {
1976
+ return typeof window !== "undefined" && !!window.sessionStorage;
1977
+ } catch {
1978
+ return false;
1979
+ }
1980
+ }, []);
1981
+ (0, react.useEffect)(() => {
1982
+ if (!isAvailable) require_api.logger_default.logError("sessionStorage is not available", void 0);
1983
+ }, [isAvailable]);
1984
+ const setItem = (0, react.useCallback)((key, value$1) => {
1985
+ if (!isAvailable) return;
1986
+ sessionStorage.setItem(key, value$1);
1987
+ window.dispatchEvent(new StorageEvent("storage", {
1988
+ key,
1989
+ newValue: value$1
1990
+ }));
1991
+ }, [isAvailable]);
1992
+ const getItem = (0, react.useCallback)((key) => {
1993
+ if (!isAvailable) return null;
1994
+ return sessionStorage.getItem(key);
1995
+ }, [isAvailable]);
1996
+ const value = (0, react.useMemo)(() => ({
1997
+ setItem,
1998
+ getItem,
1999
+ isAvailable
2000
+ }), [
2001
+ setItem,
2002
+ getItem,
2003
+ isAvailable
2004
+ ]);
2005
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(SessionStorageContext.Provider, {
2006
+ value,
2007
+ children
2008
+ });
2009
+ };
2010
+ const useSessionStorage = () => {
2011
+ const context = (0, react.useContext)(SessionStorageContext);
2012
+ if (!context) throw new Error("useSessionStorage must be used within a SessionStorageProvider");
2013
+ return context;
2014
+ };
2015
+
2016
+ //#endregion
2017
+ //#region src/hooks/useSessionStorageOperations.ts
2018
+ const useSessionStorageValue = (key) => {
2019
+ const { getItem, setItem } = useSessionStorage();
2020
+ const [value, setValue] = (0, react.useState)(() => getItem(key));
2021
+ (0, react.useEffect)(() => {
2022
+ const handleStorageChange = (event) => {
2023
+ if (event.key === key) setValue(event.newValue);
2024
+ };
2025
+ window.addEventListener("storage", handleStorageChange);
2026
+ return () => window.removeEventListener("storage", handleStorageChange);
2027
+ }, [key, getItem]);
2028
+ const updateValue = (0, react.useCallback)((newValue) => {
2029
+ setItem(key, newValue);
2030
+ setValue(newValue);
2031
+ }, [key, setItem]);
2032
+ return {
2033
+ value,
2034
+ setValue: updateValue
2035
+ };
2036
+ };
2037
+
2038
+ //#endregion
2039
+ //#region src/contexts/shopifyUrlContext.tsx
2040
+ const ShopifyUrlContext = (0, react.createContext)(void 0);
2041
+ const ShopifyUrlProvider = ({ children }) => {
2042
+ const isReady = true;
2043
+ const getTrimmedPathName = (0, react.useCallback)(() => {
2044
+ let { pathname } = window.location;
2045
+ pathname = pathname.replace("/proxy", "");
2046
+ pathname = pathname.replace(/#.*$/, "");
2047
+ pathname = pathname.replace(/\/$/, "");
2048
+ if (pathname === void 0 || pathname === null || pathname.length === 0) return null;
2049
+ return pathname;
2050
+ }, []);
2051
+ const getPlpOrPdpId = (0, react.useCallback)((extractor) => {
2052
+ if (extractor === "shopify-product-variant-id") {
2053
+ const variantId = require_graphql.parseHref(window.location.href)?.urlSearchParams?.get("variant");
2054
+ if (!variantId) return getPlpOrPdpId("shopify-product-id");
2055
+ return variantId;
2056
+ }
2057
+ const pathSegment = extractor === "shopify-product-id" ? "products" : "collections";
2058
+ const tokens = getTrimmedPathName()?.split("/");
2059
+ const idIndex = tokens?.findIndex((token) => token === pathSegment);
2060
+ if (idIndex !== void 0 && idIndex >= 0 && tokens) return decodeURIComponent(tokens[idIndex + 1]);
2061
+ return null;
2062
+ }, [getTrimmedPathName]);
2063
+ const isOnPdpPage = (0, react.useCallback)(() => {
2064
+ return getTrimmedPathName()?.includes("/products") ?? false;
2065
+ }, [getTrimmedPathName]);
2066
+ const isOnPlpPage = (0, react.useCallback)(() => {
2067
+ return (getTrimmedPathName()?.includes("/collections") && !getTrimmedPathName()?.includes("/products")) ?? false;
2068
+ }, [getTrimmedPathName]);
2069
+ const value = (0, react.useMemo)(() => ({
2070
+ getTrimmedPathName,
2071
+ getPlpOrPdpId,
2072
+ isOnPdpPage,
2073
+ isOnPlpPage,
2074
+ isReady
2075
+ }), [
2076
+ getTrimmedPathName,
2077
+ getPlpOrPdpId,
2078
+ isOnPdpPage,
2079
+ isOnPlpPage,
2080
+ isReady
2081
+ ]);
2082
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(ShopifyUrlContext.Provider, {
2083
+ value,
2084
+ children
2085
+ });
2086
+ };
2087
+ const useShopifyUrl = () => {
2088
+ const context = (0, react.useContext)(ShopifyUrlContext);
2089
+ if (!context) throw new Error("useShopifyUrl must be used within a ShopifyUrlProvider");
2090
+ return context;
2091
+ };
2092
+
2093
+ //#endregion
2094
+ //#region src/hooks/useShopifyUrlOperations.ts
2095
+ const useShopifyUrlOperations = () => {
2096
+ const { getTrimmedPathName, getPlpOrPdpId, isOnPdpPage, isOnPlpPage, isReady } = useShopifyUrl();
2097
+ return {
2098
+ getTrimmedPathName,
2099
+ getPlpOrPdpId,
2100
+ isOnPdpPage,
2101
+ isOnPlpPage,
2102
+ isReady
2103
+ };
2104
+ };
2105
+ const useCurrentPageType = () => {
2106
+ const { isOnPdpPage, isOnPlpPage, isReady } = useShopifyUrl();
2107
+ return {
2108
+ getPageType: (0, react.useCallback)(() => {
2109
+ if (!isReady) return "unknown";
2110
+ if (isOnPdpPage()) return "pdp";
2111
+ if (isOnPlpPage()) return "plp";
2112
+ return "other";
2113
+ }, [
2114
+ isReady,
2115
+ isOnPdpPage,
2116
+ isOnPlpPage
2117
+ ]),
2118
+ isReady
2119
+ };
2120
+ };
2121
+ const useProductId = (extractor) => {
2122
+ const { getPlpOrPdpId, isReady } = useShopifyUrl();
2123
+ return {
2124
+ productId: (0, react.useCallback)(() => {
2125
+ if (!isReady) return null;
2126
+ return getPlpOrPdpId(extractor);
2127
+ }, [
2128
+ isReady,
2129
+ getPlpOrPdpId,
2130
+ extractor
2131
+ ]),
2132
+ isReady
2133
+ };
2134
+ };
2135
+
2136
+ //#endregion
2137
+ //#region src/hooks/useSnapCalculator.ts
2138
+ const useSnapCalculator = (snaps, maxHeight, unit) => {
2139
+ const viewportHeightPx = document.documentElement.clientHeight;
2140
+ const swipeviewHeightPx = unit === "percent" ? Math.floor(viewportHeightPx * (maxHeight / 100)) : maxHeight;
2141
+ const snapsToPixels = (0, react.useMemo)(() => snaps?.map((snap) => Math.abs((unit === "percent" ? Math.floor(swipeviewHeightPx * (snap / 100)) : snap) - swipeviewHeightPx)), [viewportHeightPx]);
2142
+ const getPixelToSnap = (pixels) => {
2143
+ const snapIdx = snapsToPixels?.indexOf(pixels) || 0;
2144
+ return snaps?.[snapIdx] || 0;
2145
+ };
2146
+ const getSnapToPixel = (snap) => {
2147
+ const snapIdx = snaps?.indexOf(snap) || 0;
2148
+ return snapsToPixels?.[snapIdx] || 0;
2149
+ };
2150
+ return {
2151
+ viewportHeightPx,
2152
+ snapsToPixels,
2153
+ swipeviewHeightPx,
2154
+ getPixelToSnap,
2155
+ getSnapToPixel
2156
+ };
2157
+ };
2158
+
2159
+ //#endregion
2160
+ //#region src/hooks/useSystemSettingsContext.ts
2161
+ const useSystemSettingsContext = () => {
2162
+ const context = (0, react.useContext)(SystemSettingsContext);
2163
+ if (!context) throw new Error("useSystemSettingsContext must be used within a SystemSettingsContextProvider");
2164
+ return { ...context };
2165
+ };
2166
+
2167
+ //#endregion
2168
+ //#region src/hooks/useTrackComponentVisibleEvent.ts
2169
+ /**
2170
+ * Tracks a component and logs an event to Amplitude when the component is visible.
2171
+ *
2172
+ * @param component - The component to track.
2173
+ * @param element - The element to track visibility of.
2174
+ * @param eventProps - Additional properties to include with the event.
2175
+ * @param eventName - The Amplitude event name to track (defaults to ChatComponentVisible).
2176
+ */
2177
+ const useTrackComponentVisibleEvent = (component, element, eventProps, eventName = require_amplitudeContext.SpiffyMetricsEventName.ChatComponentVisible) => {
2178
+ const isVisible = useIntersection(element, "0px");
2179
+ const hasTrackedEvent = (0, react.useRef)(false);
2180
+ const { trackEvent } = require_amplitudeContext.useAmplitude();
2181
+ const componentProps = (() => {
2182
+ if (eventName === require_amplitudeContext.SpiffyMetricsEventName.ChatComponentVisible) return {
2183
+ chat_component: component,
2184
+ ...eventProps
2185
+ };
2186
+ if (eventName === require_amplitudeContext.SpiffyMetricsEventName.SearchComponentVisible) return {
2187
+ search_component: component,
2188
+ ...eventProps
2189
+ };
2190
+ return {
2191
+ component,
2192
+ ...eventProps
2193
+ };
2194
+ })();
2195
+ (0, react.useEffect)(() => {
2196
+ if (isVisible && !hasTrackedEvent.current) {
2197
+ trackEvent({
2198
+ eventName,
2199
+ eventProps: componentProps
2200
+ });
2201
+ hasTrackedEvent.current = true;
2202
+ }
2203
+ }, [
2204
+ isVisible,
2205
+ component,
2206
+ eventProps,
2207
+ eventName,
2208
+ componentProps,
2209
+ trackEvent
2210
+ ]);
2211
+ };
2212
+
2213
+ //#endregion
2214
+ //#region src/hooks/useUpdateAnalyticsProps.ts
2215
+ /**
2216
+ * Updates the default analytics properties whenever the variant info changes. This hook also
2217
+ * triggers any events that should be sent once per page visit.
2218
+ */
2219
+ const useUpdateAnalyticsProps = () => {
2220
+ const variantInfo = (0, jotai.useAtomValue)(require_app.variantInfoAtom);
2221
+ const hasInitialized = (0, react.useRef)(false);
2222
+ const hasParsedVariantInfo = (0, jotai.useAtomValue)(require_app.hasParsedVariantInfoAtom);
2223
+ const { trackEvent, setSupplementalDefaultProps } = require_amplitudeContext.useAmplitude();
2224
+ (0, react.useEffect)(() => {
2225
+ const variantInfoWithPrefix = Object.fromEntries(Object.entries(variantInfo).map(([key, value]) => [`variantInfo.${key}`, value]));
2226
+ const defaultEventProperties = {
2227
+ page_variant: variantInfo.variant,
2228
+ ...variantInfoWithPrefix
2229
+ };
2230
+ if (variantInfo.variant === "pdp") defaultEventProperties.product_id = variantInfo.productId;
2231
+ if (variantInfo.variant === "plp") defaultEventProperties.plp_id = variantInfo.plpId;
2232
+ if (variantInfo.variant === "page_visit") {
2233
+ defaultEventProperties.page_visit_category = variantInfo.pageVisitCategory;
2234
+ defaultEventProperties.page_visit_url = variantInfo.url;
2235
+ }
2236
+ setSupplementalDefaultProps(defaultEventProperties);
2237
+ if (!hasInitialized.current && hasParsedVariantInfo) {
2238
+ trackEvent({ eventName: require_amplitudeContext.SpiffyMetricsEventName.BundleLoaded });
2239
+ hasInitialized.current = true;
2240
+ }
2241
+ }, [
2242
+ variantInfo,
2243
+ hasParsedVariantInfo,
2244
+ trackEvent,
2245
+ setSupplementalDefaultProps
2246
+ ]);
2247
+ };
2248
+
2249
+ //#endregion
2250
+ //#region src/contexts/chatContext.tsx
2251
+ /**
2252
+ * Record the chat assistant response in Amplitude
2253
+ *
2254
+ * @param startTimeMs The start time of the assistant response
2255
+ * @param payload The payload used to generate the response
2256
+ */
2257
+ const recordAssistantResponse = (startTimeMs, payload, track) => {
2258
+ const atomStore = require_atomStore.getAtomStore();
2259
+ const chatState = atomStore.get(require_chat.chatAtom);
2260
+ const chatSearchState = atomStore.get(require_search.chatSearchStateAtom);
2261
+ const searchProducts = atomStore.get(require_search.chatSearchProducts);
2262
+ const searchProductsSort = atomStore.get(require_search.chatSearchProductSortingAtom);
2263
+ const assistantResponseTimeMs = {
2264
+ start: startTimeMs,
2265
+ end: Date.now()
2266
+ };
2267
+ let userQueryProperty;
2268
+ if (chatState.replyEventCategory === __spiffy_ai_commerce_api_client.UserEventCategory.SuggestionClicked && chatState.suggestion) userQueryProperty = chatState.suggestion.content;
2269
+ else if (chatState.userQuery && chatState.userQuery.length > 0) userQueryProperty = chatState.userQuery;
2270
+ const eventProps = {
2271
+ response_time_ms: assistantResponseTimeMs.end - assistantResponseTimeMs.start,
2272
+ user_event_type: chatState.replyEventCategory,
2273
+ user_query: userQueryProperty
2274
+ };
2275
+ if (chatState.replyEventCategory === __spiffy_ai_commerce_api_client.UserEventCategory.FormSubmitted) {
2276
+ const lastAssistantTurn = chatState.messages.filter((turn) => turn.length > 0 && turn[0].role === require_models.MessageRole.Assistant).pop();
2277
+ const formType = payload.userEvents?.find((event) => event.category === __spiffy_ai_commerce_api_client.UserEventCategory.FormSubmitted)?.attributes.formType;
2278
+ const formStatus = lastAssistantTurn?.some((response) => response.type === require_models.MessageType.Order);
2279
+ eventProps.form_submitted_attributes = {
2280
+ form_type: formType,
2281
+ status: formStatus ? "success" : "failed"
2282
+ };
2283
+ }
2284
+ if (chatSearchState === "product-page") {
2285
+ eventProps.search_products_returned = searchProducts.length;
2286
+ eventProps.search_products_sort_type = searchProductsSort;
2287
+ }
2288
+ track(require_amplitudeContext.SpiffyMetricsEventName.ChatAssistantResponse, { eventProps });
2289
+ };
2290
+ const ChatContext = (0, react.createContext)(void 0);
2291
+ const updateMessageState = (message, lastMessage, setMessages) => {
2292
+ if (lastMessage == null) {
2293
+ setMessages((prev) => [...prev, [message]]);
2294
+ return message;
2295
+ }
2296
+ if (lastMessage.type === require_models.MessageType.Text && message.type === require_models.MessageType.Text) {
2297
+ const newMessage = {
2298
+ ...lastMessage,
2299
+ metadata: {
2300
+ ...lastMessage.metadata,
2301
+ content: lastMessage.metadata.content + message.metadata.content
2302
+ }
2303
+ };
2304
+ setMessages((prev) => {
2305
+ const lastTurn = prev[prev.length - 1];
2306
+ return [...prev.slice(0, prev.length - 1), [...lastTurn.slice(0, lastTurn.length - 1), newMessage]];
2307
+ });
2308
+ return newMessage;
2309
+ }
2310
+ setMessages((prev) => [...prev.slice(0, prev.length - 1), [...prev[prev.length - 1], message]]);
2311
+ return message;
2312
+ };
2313
+ const handleStreamingError = (_error, setRequestFailure, setMessages) => {
2314
+ setRequestFailure(true);
2315
+ setMessages((prev) => [...prev, [{
2316
+ id: (0, uuid.v4)(),
2317
+ role: require_models.MessageRole.Assistant,
2318
+ type: require_models.MessageType.Text,
2319
+ createdAt: (/* @__PURE__ */ new Date()).toISOString(),
2320
+ metadata: { content: "I'm sorry! I'm having trouble right now. Please refresh the page or try again in a moment." }
2321
+ }]]);
2322
+ };
2323
+ const processStreamingResponse = async (stream, messageInterceptor, handleSearchResults, setMessages, setSearchIsLoading, chatId) => {
2324
+ let lastMessage;
2325
+ let hasSearchResults = false;
2326
+ for await (const response of stream) try {
2327
+ if (messageInterceptor.intercept(response)) return { hasSearchResults };
2328
+ const message = require_utils.messageFromResponse(response);
2329
+ if (!message) throw new Error("Failed to transform API response to client message");
2330
+ if (message.type === require_models.MessageType.ProductSearch) {
2331
+ handleSearchResults(message);
2332
+ hasSearchResults = true;
2333
+ setSearchIsLoading(false);
2334
+ }
2335
+ lastMessage = updateMessageState(message, lastMessage, setMessages);
2336
+ } catch (error) {
2337
+ require_api.logger_default.logWarn(`[spiffy-ai] Failed to generate responses from stream chat_id=${chatId}`, error, {
2338
+ lastResponse: lastMessage,
2339
+ response
2340
+ });
2341
+ }
2342
+ return { hasSearchResults };
2343
+ };
2344
+ const ChatContextProvider = ({ children }) => {
2345
+ const logPerfMetric = (0, jotai.useSetAtom)(require_chat.logPerfMetricAtom);
2346
+ const [widgetInitialized, setWidgetInitialized] = (0, react.useState)(false);
2347
+ const setUserHasReplied = (0, jotai.useSetAtom)(require_chat.userHasRepliedAtom);
2348
+ const [messages, setMessages] = (0, jotai.useAtom)(require_chat.messagesAtom);
2349
+ const setUserEvents = (0, jotai.useSetAtom)(require_chat.userEventsAtom);
2350
+ const setSuggestions = (0, jotai.useSetAtom)(require_chat.suggestionsAtom);
2351
+ const [suggestionsLoading, setSuggestionsLoading] = (0, jotai.useAtom)(require_chat.suggestionsLoadingAtom);
2352
+ const [responseStreaming, setResponseStreaming] = (0, jotai.useAtom)(require_chat.responseStreamingAtom);
2353
+ const setRequestFailure = (0, jotai.useSetAtom)(require_chat.requestFailureAtom);
2354
+ const userEvents = (0, jotai.useAtomValue)(require_chat.userEventQueueAtom);
2355
+ const userQueueEventCount = (0, jotai.useAtomValue)(require_chat.userQueueEventCountAtom);
2356
+ const markUserEventsProcessed = (0, jotai.useSetAtom)(require_chat.processUserEventAtom);
2357
+ const clearUserEventQueue = (0, jotai.useSetAtom)(require_chat.clearUserEventAtom);
2358
+ const userId = (0, jotai.useAtomValue)(require_app.userIdAtom);
2359
+ const chatId = (0, jotai.useAtomValue)(require_app.chatIdAtom);
2360
+ const supportedEvent = (0, jotai.useAtomValue)(require_app.supportedEventAtom);
2361
+ const orgId = "mock-org-id";
2362
+ const variantInfo = (0, jotai.useAtomValue)(require_app.variantInfoAtom);
2363
+ const settingsContext = useSystemSettingsContext();
2364
+ const messageInterceptor = require_useMessageInterceptor.useMessageInterceptor();
2365
+ const handleSearchResults = (0, jotai.useSetAtom)(require_search.handleSearchResultsAtom);
2366
+ const setSearchIsLoading = (0, jotai.useSetAtom)(require_search.chatSearchIsLoadingAtom);
2367
+ const { track } = useAmplitudeTracking();
2368
+ const getStreamingResponses = (0, react.useCallback)(async (payload) => {
2369
+ logPerfMetric(require_chat.PerfMetricsEvents.FirstResponseStarted);
2370
+ const stream = commerce_api_default.getNextResponseStreaming(payload);
2371
+ try {
2372
+ setRequestFailure(false);
2373
+ const { hasSearchResults } = await processStreamingResponse(stream, messageInterceptor, handleSearchResults, setMessages, setSearchIsLoading, chatId);
2374
+ return { hasSearchResults };
2375
+ } catch (e) {
2376
+ handleStreamingError(e, setRequestFailure, setMessages);
2377
+ throw e;
2378
+ } finally {
2379
+ logPerfMetric(require_chat.PerfMetricsEvents.FirstResponseCompleted);
2380
+ }
2381
+ }, [
2382
+ logPerfMetric,
2383
+ setRequestFailure,
2384
+ messageInterceptor,
2385
+ handleSearchResults,
2386
+ setMessages,
2387
+ setSearchIsLoading,
2388
+ chatId
2389
+ ]);
2390
+ const getSuggestions = (0, react.useCallback)(async () => {
2391
+ logPerfMetric(require_chat.PerfMetricsEvents.FirstSuggestionsStarted);
2392
+ setSuggestionsLoading(true);
2393
+ setSuggestions([]);
2394
+ const payloadWithoutAppLoaded = require_chat.createResponsePayload({
2395
+ userEvents: [],
2396
+ generationParams: settingsContext.generationParams
2397
+ });
2398
+ const response = await commerce_api_default.getNextSuggestions(payloadWithoutAppLoaded);
2399
+ setSuggestions(response.sort((a, b) => a.content.length - b.content.length));
2400
+ setSuggestionsLoading(false);
2401
+ logPerfMetric(require_chat.PerfMetricsEvents.FirstSuggestionsCompleted);
2402
+ }, [
2403
+ logPerfMetric,
2404
+ setSuggestionsLoading,
2405
+ setSuggestions,
2406
+ settingsContext.generationParams
2407
+ ]);
2408
+ const getResponses = (0, react.useCallback)(async (payload) => {
2409
+ try {
2410
+ const requestPayload = payload ?? require_chat.createResponsePayload({
2411
+ userEvents,
2412
+ generationParams: settingsContext.generationParams
2413
+ });
2414
+ setResponseStreaming(true);
2415
+ setSuggestions([]);
2416
+ const startTimeMs = Date.now();
2417
+ await getStreamingResponses(requestPayload);
2418
+ recordAssistantResponse(startTimeMs, requestPayload, track);
2419
+ await getSuggestions();
2420
+ } catch (error) {
2421
+ require_api.logger_default.logError("[spiffy-ai] getResponses error", error);
2422
+ } finally {
2423
+ markUserEventsProcessed(userEvents.map(({ eventId }) => eventId));
2424
+ setUserHasReplied(false);
2425
+ setResponseStreaming(false);
2426
+ }
2427
+ }, [
2428
+ userEvents,
2429
+ settingsContext.generationParams,
2430
+ setResponseStreaming,
2431
+ setSuggestions,
2432
+ getStreamingResponses,
2433
+ markUserEventsProcessed,
2434
+ getSuggestions,
2435
+ setUserHasReplied,
2436
+ track
2437
+ ]);
2438
+ (0, react.useEffect)(() => {
2439
+ const processUserEvents = async () => {
2440
+ if (responseStreaming || !widgetInitialized) return;
2441
+ if (variantInfo.variant === "pdp" && !variantInfo.productId || variantInfo.variant === "plp" && !variantInfo.plpId || variantInfo.variant === "page_visit" && !variantInfo.url) {
2442
+ require_api.logger_default.logDebug("[spiffy-ai] variantInfo has invalid values, skipping...", {
2443
+ variantInfo,
2444
+ supportedEvent
2445
+ });
2446
+ return;
2447
+ }
2448
+ require_api.logger_default.logDebug(`Assistants Turn is_currently_streaming=${responseStreaming} initialized=${widgetInitialized}`);
2449
+ try {
2450
+ await getResponses();
2451
+ require_api.logger_default.logInfo(`Assistants Turn [finished]`);
2452
+ } catch (error) {
2453
+ require_api.logger_default.logError("[spiffy-ai] Assistants Turn error", error);
2454
+ }
2455
+ };
2456
+ if (userQueueEventCount > 0) processUserEvents();
2457
+ }, [
2458
+ getResponses,
2459
+ responseStreaming,
2460
+ userQueueEventCount,
2461
+ widgetInitialized,
2462
+ variantInfo,
2463
+ supportedEvent
2464
+ ]);
2465
+ (0, react.useEffect)(() => {
2466
+ if (widgetInitialized || responseStreaming) {
2467
+ require_api.logger_default.logDebug(`[spiffy-ai] initializeWidget [skipped] is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`);
2468
+ return;
2469
+ }
2470
+ const hydrateChat = async () => {
2471
+ try {
2472
+ require_api.logger_default.logDebug(`[spiffy-ai] initializeWidget is_currently_streaming=${responseStreaming} is_initialized=${widgetInitialized}`);
2473
+ const { messages: existingMessages, userEvents: userEvents$1 } = await commerce_api_default.getResponses(orgId, chatId, userId);
2474
+ setMessages([...existingMessages]);
2475
+ setUserEvents([...userEvents$1]);
2476
+ getResponses();
2477
+ } catch (error) {
2478
+ require_api.logger_default.logInfo(`Init chat [exception] chat_id=${chatId} error=${error}`, error);
2479
+ if (error instanceof SessionRestartRequired) {
2480
+ const appLoadedEvent = createAppLoadedEvent();
2481
+ const visitEvent = createVisitUserEvent({ variantInfo });
2482
+ setMessages([]);
2483
+ clearUserEventQueue();
2484
+ if (visitEvent) {
2485
+ const payload = require_chat.createResponsePayload({
2486
+ userEvents: [appLoadedEvent, visitEvent],
2487
+ generationParams: settingsContext.generationParams
2488
+ });
2489
+ getResponses(payload);
2490
+ }
2491
+ }
2492
+ } finally {
2493
+ setWidgetInitialized(true);
2494
+ }
2495
+ };
2496
+ hydrateChat();
2497
+ }, []);
2498
+ const onFocus = (0, react.useCallback)(async () => {
2499
+ try {
2500
+ if (!responseStreaming && !suggestionsLoading && orgId) {
2501
+ const { messages: existingMessages } = await commerce_api_default.getResponses(orgId, chatId, userId);
2502
+ if (existingMessages.length > messages.length) setMessages([...existingMessages]);
2503
+ }
2504
+ } catch (error) {
2505
+ require_api.logger_default.logError("[spiffy-ai] onFocus error", error);
2506
+ }
2507
+ }, [
2508
+ responseStreaming,
2509
+ suggestionsLoading,
2510
+ orgId,
2511
+ chatId,
2512
+ userId,
2513
+ messages.length,
2514
+ setMessages
2515
+ ]);
2516
+ (0, react.useEffect)(() => {
2517
+ window.addEventListener("focus", onFocus);
2518
+ return () => {
2519
+ window.removeEventListener("focus", onFocus);
2520
+ };
2521
+ }, [onFocus]);
2522
+ const chatContext = (0, react.useMemo)(() => ({}), []);
2523
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(ChatContext.Provider, {
2524
+ value: chatContext,
2525
+ children
2526
+ });
2527
+ };
2528
+
2529
+ //#endregion
2530
+ //#region src/contexts/enviveCssContext.tsx
2531
+ const EnviveCssProvider = ({ children }) => {
2532
+ const { colorsConfig, frontendConfig, loading } = useNewOrgConfig();
2533
+ let merchantThemeCss = `* {}`;
2534
+ if (colorsConfig && !loading) merchantThemeCss = `
2535
+ * {
2536
+ --spiffy-colors-text-primary: ${colorsConfig.textPrimary};
2537
+ --spiffy-colors-text-secondary: ${colorsConfig.textSecondary};
2538
+ --spiffy-colors-text-accent: ${colorsConfig.textAccent};
2539
+ --spiffy-colors-text-link: ${colorsConfig.textLink};
2540
+ --spiffy-colors-text-light: ${colorsConfig.textLight};
2541
+ --spiffy-colors-background-primary: ${colorsConfig.backgroundPrimary};
2542
+ --spiffy-colors-background-secondary: ${colorsConfig.backgroundSecondary};
2543
+ --spiffy-colors-background-secondary-dark: ${colorsConfig.backgroundSecondaryDark};
2544
+ --spiffy-colors-background-tertiary: ${colorsConfig.backgroundTertiary};
2545
+ --spiffy-colors-background-dark: ${colorsConfig.backgroundDark};
2546
+ --spiffy-colors-background-light: ${colorsConfig.backgroundLight};
2547
+ --spiffy-colors-background-saturated: ${colorsConfig.backgroundSaturated};
2548
+ --spiffy-colors-border-light: ${colorsConfig.borderLight};
2549
+ --spiffy-colors-border-medium: ${colorsConfig.borderMedium};
2550
+ --spiffy-colors-border-dark: ${colorsConfig.borderDark};
2551
+ --spiffy-colors-border-outline: ${colorsConfig.borderOutline};
2552
+ --spiffy-colors-accent-primary: ${colorsConfig.accentPrimary};
2553
+ --spiffy-colors-accent-secondary: ${colorsConfig.accentSecondary};
2554
+ }`;
2555
+ console.log(frontendConfig);
2556
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsxs)(react_jsx_runtime.Fragment, { children: [
2557
+ frontendConfig ? /* @__PURE__ */ (0, react_jsx_runtime.jsx)("style", {
2558
+ id: "merchant-css-overrides",
2559
+ children: `${frontendConfig.merchantOverrideCss}`
2560
+ }) : null,
2561
+ colorsConfig ? /* @__PURE__ */ (0, react_jsx_runtime.jsx)("style", {
2562
+ id: "merchant-css-colors",
2563
+ children: `${merchantThemeCss}`
2564
+ }) : null,
2565
+ children
2566
+ ] });
2567
+ };
2568
+
2569
+ //#endregion
2570
+ //#region src/contexts/systemSettingsContext.tsx
2571
+ const getChatModelName = () => {
2572
+ const urlObj = new URL(window.location.href);
2573
+ const params = new URLSearchParams(urlObj.search);
2574
+ return Object.fromEntries(params.entries()).llm_model_name;
2575
+ };
2576
+ const defaultGenerationParams = {
2577
+ stream: true,
2578
+ numSuggestions: 3,
2579
+ model: getChatModelName()
2580
+ };
2581
+ const SystemSettingsContext = (0, react.createContext)(void 0);
2582
+ const SystemSettingsContextProvider = ({ children, generationParams, showDebugBar }) => {
2583
+ const [params, setParams] = (0, react.useState)(() => generationParams ?? defaultGenerationParams);
2584
+ const endpointURL = (0, jotai.useAtomValue)(require_enviveConfig.baseUrlAtom);
2585
+ const settingsContextValues = (0, react.useMemo)(() => ({
2586
+ generationParams: params,
2587
+ showDebugBar,
2588
+ setGenerationParams: setParams,
2589
+ endpointURL
2590
+ }), [
2591
+ generationParams,
2592
+ endpointURL,
2593
+ showDebugBar
2594
+ ]);
2595
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(SystemSettingsContext.Provider, {
2596
+ value: settingsContextValues,
2597
+ children
2598
+ });
2599
+ };
2600
+
2601
+ //#endregion
2602
+ Object.defineProperty(exports, 'CdnProvider', {
2603
+ enumerable: true,
2604
+ get: function () {
2605
+ return CdnProvider;
2606
+ }
2607
+ });
2608
+ Object.defineProperty(exports, 'ChatContext', {
2609
+ enumerable: true,
2610
+ get: function () {
2611
+ return ChatContext;
2612
+ }
2613
+ });
2614
+ Object.defineProperty(exports, 'ChatContextProvider', {
2615
+ enumerable: true,
2616
+ get: function () {
2617
+ return ChatContextProvider;
2618
+ }
2619
+ });
2620
+ Object.defineProperty(exports, 'EnviveCssProvider', {
2621
+ enumerable: true,
2622
+ get: function () {
2623
+ return EnviveCssProvider;
2624
+ }
2625
+ });
2626
+ Object.defineProperty(exports, 'FeatureFlagServiceProvider', {
2627
+ enumerable: true,
2628
+ get: function () {
2629
+ return FeatureFlagServiceProvider;
2630
+ }
2631
+ });
2632
+ Object.defineProperty(exports, 'GraphQLProvider', {
2633
+ enumerable: true,
2634
+ get: function () {
2635
+ return GraphQLProvider;
2636
+ }
2637
+ });
2638
+ Object.defineProperty(exports, 'NewOrgConfigProvider', {
2639
+ enumerable: true,
2640
+ get: function () {
2641
+ return NewOrgConfigProvider;
2642
+ }
2643
+ });
2644
+ Object.defineProperty(exports, 'SearchProvider', {
2645
+ enumerable: true,
2646
+ get: function () {
2647
+ return SearchProvider;
2648
+ }
2649
+ });
2650
+ Object.defineProperty(exports, 'SearchResultsState', {
2651
+ enumerable: true,
2652
+ get: function () {
2653
+ return SearchResultsState;
2654
+ }
2655
+ });
2656
+ Object.defineProperty(exports, 'SessionStorageProvider', {
2657
+ enumerable: true,
2658
+ get: function () {
2659
+ return SessionStorageProvider;
2660
+ }
2661
+ });
2662
+ Object.defineProperty(exports, 'ShopifyUrlProvider', {
2663
+ enumerable: true,
2664
+ get: function () {
2665
+ return ShopifyUrlProvider;
2666
+ }
2667
+ });
2668
+ Object.defineProperty(exports, 'SystemSettingsContext', {
2669
+ enumerable: true,
2670
+ get: function () {
2671
+ return SystemSettingsContext;
2672
+ }
2673
+ });
2674
+ Object.defineProperty(exports, 'SystemSettingsContextProvider', {
2675
+ enumerable: true,
2676
+ get: function () {
2677
+ return SystemSettingsContextProvider;
2678
+ }
2679
+ });
2680
+ Object.defineProperty(exports, 'UserIdentityProvider', {
2681
+ enumerable: true,
2682
+ get: function () {
2683
+ return UserIdentityProvider;
2684
+ }
2685
+ });
2686
+ Object.defineProperty(exports, 'createAppLoadedEvent', {
2687
+ enumerable: true,
2688
+ get: function () {
2689
+ return createAppLoadedEvent;
2690
+ }
2691
+ });
2692
+ Object.defineProperty(exports, 'createVisitUserEvent', {
2693
+ enumerable: true,
2694
+ get: function () {
2695
+ return createVisitUserEvent;
2696
+ }
2697
+ });
2698
+ Object.defineProperty(exports, 'defaultGenerationParams', {
2699
+ enumerable: true,
2700
+ get: function () {
2701
+ return defaultGenerationParams;
2702
+ }
2703
+ });
2704
+ Object.defineProperty(exports, 'getSearchResultsState', {
2705
+ enumerable: true,
2706
+ get: function () {
2707
+ return getSearchResultsState;
2708
+ }
2709
+ });
2710
+ Object.defineProperty(exports, 'isElementPartiallyVisible', {
2711
+ enumerable: true,
2712
+ get: function () {
2713
+ return isElementPartiallyVisible;
2714
+ }
2715
+ });
2716
+ Object.defineProperty(exports, 'isWithinBusinessHours', {
2717
+ enumerable: true,
2718
+ get: function () {
2719
+ return isWithinBusinessHours;
2720
+ }
2721
+ });
2722
+ Object.defineProperty(exports, 'useAmplitudeTracking', {
2723
+ enumerable: true,
2724
+ get: function () {
2725
+ return useAmplitudeTracking;
2726
+ }
2727
+ });
2728
+ Object.defineProperty(exports, 'useAssetUrl', {
2729
+ enumerable: true,
2730
+ get: function () {
2731
+ return useAssetUrl;
2732
+ }
2733
+ });
2734
+ Object.defineProperty(exports, 'useBlockBackButton', {
2735
+ enumerable: true,
2736
+ get: function () {
2737
+ return useBlockBackButton;
2738
+ }
2739
+ });
2740
+ Object.defineProperty(exports, 'useCdn', {
2741
+ enumerable: true,
2742
+ get: function () {
2743
+ return useCdn;
2744
+ }
2745
+ });
2746
+ Object.defineProperty(exports, 'useCdnBasePath', {
2747
+ enumerable: true,
2748
+ get: function () {
2749
+ return useCdnBasePath;
2750
+ }
2751
+ });
2752
+ Object.defineProperty(exports, 'useCdnUrl', {
2753
+ enumerable: true,
2754
+ get: function () {
2755
+ return useCdnUrl;
2756
+ }
2757
+ });
2758
+ Object.defineProperty(exports, 'useChatToggle', {
2759
+ enumerable: true,
2760
+ get: function () {
2761
+ return useChatToggle;
2762
+ }
2763
+ });
2764
+ Object.defineProperty(exports, 'useChatToggleAnalytics', {
2765
+ enumerable: true,
2766
+ get: function () {
2767
+ return useChatToggleAnalytics;
2768
+ }
2769
+ });
2770
+ Object.defineProperty(exports, 'useColorsAndFrontendConfig', {
2771
+ enumerable: true,
2772
+ get: function () {
2773
+ return useColorsAndFrontendConfig;
2774
+ }
2775
+ });
2776
+ Object.defineProperty(exports, 'useCurrentPageType', {
2777
+ enumerable: true,
2778
+ get: function () {
2779
+ return useCurrentPageType;
2780
+ }
2781
+ });
2782
+ Object.defineProperty(exports, 'useCustomerSupportHandoff', {
2783
+ enumerable: true,
2784
+ get: function () {
2785
+ return useCustomerSupportHandoff;
2786
+ }
2787
+ });
2788
+ Object.defineProperty(exports, 'useDebounce', {
2789
+ enumerable: true,
2790
+ get: function () {
2791
+ return useDebounce;
2792
+ }
2793
+ });
2794
+ Object.defineProperty(exports, 'useElementObserver', {
2795
+ enumerable: true,
2796
+ get: function () {
2797
+ return useElementObserver;
2798
+ }
2799
+ });
2800
+ Object.defineProperty(exports, 'useEnviveFeatureFlag', {
2801
+ enumerable: true,
2802
+ get: function () {
2803
+ return useEnviveFeatureFlag;
2804
+ }
2805
+ });
2806
+ Object.defineProperty(exports, 'useFeatureFlagService', {
2807
+ enumerable: true,
2808
+ get: function () {
2809
+ return useFeatureFlagService;
2810
+ }
2811
+ });
2812
+ Object.defineProperty(exports, 'useGrabAndScroll', {
2813
+ enumerable: true,
2814
+ get: function () {
2815
+ return useGrabAndScroll;
2816
+ }
2817
+ });
2818
+ Object.defineProperty(exports, 'useGraphQLClient', {
2819
+ enumerable: true,
2820
+ get: function () {
2821
+ return useGraphQLClient;
2822
+ }
2823
+ });
2824
+ Object.defineProperty(exports, 'useIdentifyUser', {
2825
+ enumerable: true,
2826
+ get: function () {
2827
+ return useIdentifyUser;
2828
+ }
2829
+ });
2830
+ Object.defineProperty(exports, 'useImageResolver', {
2831
+ enumerable: true,
2832
+ get: function () {
2833
+ return useImageResolver;
2834
+ }
2835
+ });
2836
+ Object.defineProperty(exports, 'useIntersection', {
2837
+ enumerable: true,
2838
+ get: function () {
2839
+ return useIntersection;
2840
+ }
2841
+ });
2842
+ Object.defineProperty(exports, 'useIsSmallScreen', {
2843
+ enumerable: true,
2844
+ get: function () {
2845
+ return useIsSmallScreen;
2846
+ }
2847
+ });
2848
+ Object.defineProperty(exports, 'useLocalStorageListener', {
2849
+ enumerable: true,
2850
+ get: function () {
2851
+ return useLocalStorageListener;
2852
+ }
2853
+ });
2854
+ Object.defineProperty(exports, 'useLocalStorageValue', {
2855
+ enumerable: true,
2856
+ get: function () {
2857
+ return useLocalStorageValue;
2858
+ }
2859
+ });
2860
+ Object.defineProperty(exports, 'useMessageFilter', {
2861
+ enumerable: true,
2862
+ get: function () {
2863
+ return useMessageFilter;
2864
+ }
2865
+ });
2866
+ Object.defineProperty(exports, 'useMessageScrollObserver', {
2867
+ enumerable: true,
2868
+ get: function () {
2869
+ return useMessageScrollObserver;
2870
+ }
2871
+ });
2872
+ Object.defineProperty(exports, 'useNewOrgConfig', {
2873
+ enumerable: true,
2874
+ get: function () {
2875
+ return useNewOrgConfig;
2876
+ }
2877
+ });
2878
+ Object.defineProperty(exports, 'useNewOrgConfigContext', {
2879
+ enumerable: true,
2880
+ get: function () {
2881
+ return useNewOrgConfigContext;
2882
+ }
2883
+ });
2884
+ Object.defineProperty(exports, 'useOrgId', {
2885
+ enumerable: true,
2886
+ get: function () {
2887
+ return useOrgId;
2888
+ }
2889
+ });
2890
+ Object.defineProperty(exports, 'useProductId', {
2891
+ enumerable: true,
2892
+ get: function () {
2893
+ return useProductId;
2894
+ }
2895
+ });
2896
+ Object.defineProperty(exports, 'useProductSearch', {
2897
+ enumerable: true,
2898
+ get: function () {
2899
+ return useProductSearch;
2900
+ }
2901
+ });
2902
+ Object.defineProperty(exports, 'useSearch', {
2903
+ enumerable: true,
2904
+ get: function () {
2905
+ return useSearch;
2906
+ }
2907
+ });
2908
+ Object.defineProperty(exports, 'useSearchService', {
2909
+ enumerable: true,
2910
+ get: function () {
2911
+ return useSearchService;
2912
+ }
2913
+ });
2914
+ Object.defineProperty(exports, 'useSearchWithQuery', {
2915
+ enumerable: true,
2916
+ get: function () {
2917
+ return useSearchWithQuery;
2918
+ }
2919
+ });
2920
+ Object.defineProperty(exports, 'useSessionStorage', {
2921
+ enumerable: true,
2922
+ get: function () {
2923
+ return useSessionStorage;
2924
+ }
2925
+ });
2926
+ Object.defineProperty(exports, 'useSessionStorageValue', {
2927
+ enumerable: true,
2928
+ get: function () {
2929
+ return useSessionStorageValue;
2930
+ }
2931
+ });
2932
+ Object.defineProperty(exports, 'useShopifyUrl', {
2933
+ enumerable: true,
2934
+ get: function () {
2935
+ return useShopifyUrl;
2936
+ }
2937
+ });
2938
+ Object.defineProperty(exports, 'useShopifyUrlOperations', {
2939
+ enumerable: true,
2940
+ get: function () {
2941
+ return useShopifyUrlOperations;
2942
+ }
2943
+ });
2944
+ Object.defineProperty(exports, 'useSnapCalculator', {
2945
+ enumerable: true,
2946
+ get: function () {
2947
+ return useSnapCalculator;
2948
+ }
2949
+ });
2950
+ Object.defineProperty(exports, 'useSpiffyFeatureFlag', {
2951
+ enumerable: true,
2952
+ get: function () {
2953
+ return useSpiffyFeatureFlag;
2954
+ }
2955
+ });
2956
+ Object.defineProperty(exports, 'useSystemSettingsContext', {
2957
+ enumerable: true,
2958
+ get: function () {
2959
+ return useSystemSettingsContext;
2960
+ }
2961
+ });
2962
+ Object.defineProperty(exports, 'useTrackComponentVisibleEvent', {
2963
+ enumerable: true,
2964
+ get: function () {
2965
+ return useTrackComponentVisibleEvent;
2966
+ }
2967
+ });
2968
+ Object.defineProperty(exports, 'useUpdateAnalyticsProps', {
2969
+ enumerable: true,
2970
+ get: function () {
2971
+ return useUpdateAnalyticsProps;
2972
+ }
2973
+ });
2974
+ Object.defineProperty(exports, 'useUserIdentity', {
2975
+ enumerable: true,
2976
+ get: function () {
2977
+ return useUserIdentity;
2978
+ }
2979
+ });
2980
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"contexts-B4ihTBsV.cjs","names":["CdnProvider: React.FC<{ children: React.ReactNode }>","cdnUrlAtom","errorResponseBody","throwSessionRestartRequiredIf","ResponseError","getAtomStore","baseUrlAtom","config: Configuration","Configuration","DefaultApi","InferenceApi","CustomerServiceApi","orgShortNameAtom","orgIdAtom","userIdAtom","chatIdAtom","contextSourceAtom","envAtom","featureFlagServiceAtom","context: Context","ContextSourceEnum","ContextEnvEnum","messageRequestToCommerceMessageRequest","validateResponse","messageFromResponse","err: unknown","error: unknown","validateSuggestion","data: V1GetSessionMessages200Response","responses: Response[][]","suggestions: Suggestion[]","userEvents: UserEvent[]","validateUserEvent","UserEventCategory","assistantMessages: Message[][]","ResponseCategory","userMessages: Message[][]","messageFromQueryEvent","messageFromSuggestionEvent","FormType","messageFromFormSubmittedEvent","coreSupportedEventRequestToApiRequest","V1OrgConfigGetSourceEnum","reactAppNameAtom","request: V1OrgConfigGetRequest","ProductExperiment","validateOrgConfigResults","useAmplitude","chatOnToggleAtom","chatAtom","useAmplitude","SpiffyMetricsEventName","chatOnToggleAtom","DOMObserver","configVersion","orgLevelApiKeyAtom","baseUrlAtom","getMerchantOrgIdQuery","validateGraphQLOrgId","transformSnakeToCamel","UAParser","UserIdentityProvider: React.FC<{ children: React.ReactNode }>","useLocalStorage","CommerceApiClient","OrgShortName","orgShortNameAtom","useLocalStorage","listener: LocalStorageEventListener","LocalStorageKeys","MessageType","FeatureGates","FeatureFlagServiceProvider: React.FC<\n  FeatureFlagServiceProviderProps\n>","NewOrgConfigProvider: React.FC<NewOrgConfigProviderProps>","orgShortNameAtom","newOrgConfigAtom","getAtomStore","orgIdAtom","UserEventCategory","PLPAttributeCategory","orgShortNameAtom","searchAtom","filteredSearchProductsAtom","performSearchAtom","autocompleteStateAtom","searchParamsAtom","isFilterOpenAtom","searchSelectedFiltersAtom","addSearchFilterAtom","removeSearchFilterAtom","searchProductSortingAtom","clearSearchFiltersAtom","searchFiltersAtom","formatFilterDisplayName","ProductSorting","useAmplitude","SpiffyMetricsEventName","createFilterOption","SpiffyWidgets","handleSelectFilterItem: SelectFilterItem","enviveOrgShortNameAtom","chatIdAtom","userIdAtom","contextSourceAtom","ContextSourceEnum","enviveEnvAtom","ContextEnvEnum","variantInfoAtom","ResponseCategory","ResponseError","SearchProvider: React.FC<{ children: React.ReactNode }>","useEnviveConfig","baseUrlAtom","config: Configuration","Configuration","SearchApi","SessionStorageProvider: React.FC<{\n  children: React.ReactNode;\n}>","value","ShopifyUrlProvider: React.FC<{ children: React.ReactNode }>","parseHref","SpiffyMetricsEventName","useAmplitude","variantInfoAtom","hasParsedVariantInfoAtom","useAmplitude","defaultEventProperties: Record<string, unknown>","SpiffyMetricsEventName","getAtomStore","chatAtom","chatSearchStateAtom","chatSearchProducts","chatSearchProductSortingAtom","userQueryProperty: string | undefined","UserEventCategory","eventProps: Record<string, unknown>","MessageRole","MessageType","SpiffyMetricsEventName","lastMessage: Message | undefined","messageFromResponse","error: unknown","logPerfMetricAtom","userHasRepliedAtom","messagesAtom","userEventsAtom","suggestionsAtom","suggestionsLoadingAtom","responseStreamingAtom","requestFailureAtom","userEventQueueAtom","userQueueEventCountAtom","processUserEventAtom","clearUserEventAtom","userIdAtom","chatIdAtom","supportedEventAtom","variantInfoAtom","useMessageInterceptor","handleSearchResultsAtom","chatSearchIsLoadingAtom","PerfMetricsEvents","CommerceApiClient","createResponsePayload","userEvents","EnviveCssProvider: React.FC<EnviveCssProviderProps>","defaultGenerationParams: GenerationParams","baseUrlAtom"],"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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"}