@envive-ai/react-hooks 0.1.0

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 (385) hide show
  1. package/LICENSE +2 -0
  2. package/README.md +2 -0
  3. package/dist/GridInsertionService-CEYo9pGj.js +22 -0
  4. package/dist/GridInsertionService-CS_bnPh0.cjs +28 -0
  5. package/dist/bandolier-Ble8jEa8.js +1221 -0
  6. package/dist/bandolier-Bm2xAt_j.cjs +1221 -0
  7. package/dist/carpe-Da7b-LCW.cjs +599 -0
  8. package/dist/carpe-W13mhRRP.js +597 -0
  9. package/dist/cdnService-CZ-aXcY6.cjs +23 -0
  10. package/dist/cdnService-zQfKk3Eb.js +18 -0
  11. package/dist/chatElementDisplayLocation-CX8fuNao.d.cts +239 -0
  12. package/dist/chatElementDisplayLocation-CwptS9tx.d.ts +239 -0
  13. package/dist/chunk-CUT6urMc.cjs +30 -0
  14. package/dist/contexts/index.cjs +13 -0
  15. package/dist/contexts/index.d.cts +65 -0
  16. package/dist/contexts/index.d.ts +66 -0
  17. package/dist/contexts/index.js +7 -0
  18. package/dist/contexts-BRjfVq_k.js +5064 -0
  19. package/dist/contexts-BYArqZtK.cjs +5164 -0
  20. package/dist/coterie-3y0D9ko4.cjs +229 -0
  21. package/dist/coterie-DOWcJAYv.js +229 -0
  22. package/dist/custservice-types-CFIFwZ-r.js +10 -0
  23. package/dist/custservice-types-CkfxZiHY.cjs +16 -0
  24. package/dist/default-C2fEZKXk.js +175 -0
  25. package/dist/default-CBUq6Q6G.cjs +4 -0
  26. package/dist/default-CGIFZK6m.js +4 -0
  27. package/dist/default-D_KPZdPJ.cjs +198 -0
  28. package/dist/divIds-Bss-btao.js +49 -0
  29. package/dist/divIds-DnZNd7rA.cjs +223 -0
  30. package/dist/dreamlandBaby-DCIsuU9R.cjs +338 -0
  31. package/dist/dreamlandBaby-DvSaZGrz.js +338 -0
  32. package/dist/entrypoints-D_JUvkgy.cjs +18 -0
  33. package/dist/entrypoints-YLQsbBRD.js +6 -0
  34. package/dist/enviveConfigContext-CUGLpPGU.js +34 -0
  35. package/dist/enviveConfigContext-Dfr2VH6u.cjs +48 -0
  36. package/dist/fiveCbd-B1SESMCO.js +605 -0
  37. package/dist/fiveCbd-CkOlVby_.cjs +605 -0
  38. package/dist/forLoveAndLemons-CfYPMnKS.cjs +660 -0
  39. package/dist/forLoveAndLemons-DmwYZIk0.js +658 -0
  40. package/dist/greenpan-Bsl3ir59.cjs +389 -0
  41. package/dist/greenpan-BtOi45lf.js +389 -0
  42. package/dist/grooveLife-6_dtYsRk.js +334 -0
  43. package/dist/grooveLife-Cmm1PSCL.cjs +334 -0
  44. package/dist/homegrownCannabis-C-kw-74X.js +400 -0
  45. package/dist/homegrownCannabis-CO0uY_mp.cjs +400 -0
  46. package/dist/hooks/index.cjs +16 -0
  47. package/dist/hooks/index.d.cts +357 -0
  48. package/dist/hooks/index.d.ts +357 -0
  49. package/dist/hooks/index.js +7 -0
  50. package/dist/jackArcher-CLVmwwpI.js +719 -0
  51. package/dist/jackArcher-DdYTIzAV.cjs +719 -0
  52. package/dist/jordanCraig-Am-Oor-O.js +1778 -0
  53. package/dist/jordanCraig-_u3-w4Hp.cjs +1778 -0
  54. package/dist/kindredBravely-CWovIDSc.cjs +482 -0
  55. package/dist/kindredBravely-eWp-ud_E.js +482 -0
  56. package/dist/kutFromTheKloth-BMV4BuGQ.js +361 -0
  57. package/dist/kutFromTheKloth-Q589bAOC.cjs +361 -0
  58. package/dist/larryAndSerges-BMUlTgI-.js +252 -0
  59. package/dist/larryAndSerges-CEau764j.cjs +252 -0
  60. package/dist/leapsAndRebounds-DGMzPO7T.js +352 -0
  61. package/dist/leapsAndRebounds-DHAtRTJD.cjs +352 -0
  62. package/dist/logger-Dln20ans.cjs +25 -0
  63. package/dist/logger-pdEEY8T2.js +19 -0
  64. package/dist/longevityrx-CZW8Hxzi.cjs +312 -0
  65. package/dist/longevityrx-jH2JLhNH.js +312 -0
  66. package/dist/lookOptic-BGXP5P_V.js +274 -0
  67. package/dist/lookOptic-CA6RwLbG.cjs +274 -0
  68. package/dist/mantraBrand-Cm9_PBCT.js +742 -0
  69. package/dist/mantraBrand-DByNqpnL.cjs +742 -0
  70. package/dist/medterra-B0wxj_PV.js +575 -0
  71. package/dist/medterra-DnPN2ksU.cjs +575 -0
  72. package/dist/modells-Bmz8Ag5M.js +476 -0
  73. package/dist/modells-CoYgkLSp.cjs +476 -0
  74. package/dist/models-DHdb7QWn.js +51 -0
  75. package/dist/models-ixxUsGL_.cjs +69 -0
  76. package/dist/pressedFloral-DSKs_oVG.js +653 -0
  77. package/dist/pressedFloral-DjBiSoUl.cjs +653 -0
  78. package/dist/skinPerfection-B_3xzVNS.cjs +326 -0
  79. package/dist/skinPerfection-IDrBuAPt.js +326 -0
  80. package/dist/snapSupplements-BJk5T5ba.js +277 -0
  81. package/dist/snapSupplements-BStTsdOZ.cjs +277 -0
  82. package/dist/socialProofClasses-Bhv2Vulz.js +9 -0
  83. package/dist/socialProofClasses-CrQBWdSA.cjs +39 -0
  84. package/dist/spanx-BYg0LE7R.js +653 -0
  85. package/dist/spanx-LwU1zSzq.cjs +655 -0
  86. package/dist/spanxStaging-CfSmuKYB.js +837 -0
  87. package/dist/spanxStaging-OZLV9qix.cjs +840 -0
  88. package/dist/suggestionBarV2-types-BllzwsBD.js +34 -0
  89. package/dist/suggestionBarV2-types-CaovchMP.cjs +46 -0
  90. package/dist/supergoop-BqPXDnKk.cjs +327 -0
  91. package/dist/supergoop-CIlrHND_.js +325 -0
  92. package/dist/types-C4T5UOIW.cjs +230 -0
  93. package/dist/types-CYNvLeSA.js +176 -0
  94. package/dist/uniqueVintage-B30mOqbH.cjs +1205 -0
  95. package/dist/uniqueVintage-CFueJOhO.js +1203 -0
  96. package/dist/venaCbd-DHGZy49P.cjs +357 -0
  97. package/dist/venaCbd-T0CqVD4k.js +357 -0
  98. package/dist/westonJonBoucher-BdMzs_Yg.cjs +414 -0
  99. package/dist/westonJonBoucher-b4TCQ4ev.js +414 -0
  100. package/dist/wineEnthusiast-BLGlOjgr.cjs +932 -0
  101. package/dist/wineEnthusiast-BqR0i_54.js +932 -0
  102. package/dist/wolfMattress-CyyO-LoC.js +362 -0
  103. package/dist/wolfMattress-DNGZOivg.cjs +362 -0
  104. package/dist/wolfTactical-3Mm2fvVF.js +341 -0
  105. package/dist/wolfTactical-BmXYlFjr.cjs +341 -0
  106. package/package.json +66 -0
  107. package/src/adapters/amplitude/amplitudeAdapter.ts +454 -0
  108. package/src/adapters/amplitude/index.ts +2 -0
  109. package/src/adapters/amplitude/stubAmplitudeAdapter.ts +34 -0
  110. package/src/adapters/spiffy/commerce/api.ts +596 -0
  111. package/src/adapters/spiffy/commerce/exceptions/sessionExceptions.ts +6 -0
  112. package/src/adapters/spiffy/commerce/exceptions/unsupportedProductExceptions.ts +6 -0
  113. package/src/adapters/spiffy/commerce/graphql.ts +184 -0
  114. package/src/application/config/generalStaticConfig.ts +37 -0
  115. package/src/application/logging/logger.ts +29 -0
  116. package/src/application/models/api/context.ts +4 -0
  117. package/src/application/models/api/generationParams.ts +4 -0
  118. package/src/application/models/api/nextMessageRequest.ts +11 -0
  119. package/src/application/models/api/orgAnalyticsConfig.ts +19 -0
  120. package/src/application/models/api/orgConfigResults.ts +40 -0
  121. package/src/application/models/api/organizationConfig.ts +12 -0
  122. package/src/application/models/api/response.ts +132 -0
  123. package/src/application/models/api/responseGenerics.ts +67 -0
  124. package/src/application/models/api/search.ts +26 -0
  125. package/src/application/models/api/suggestion.ts +4 -0
  126. package/src/application/models/api/supportedEventRequest.ts +8 -0
  127. package/src/application/models/api/userEvent.ts +101 -0
  128. package/src/application/models/cachedValue.ts +8 -0
  129. package/src/application/models/chatElementDisplayLocation.ts +22 -0
  130. package/src/application/models/clientDetails.ts +18 -0
  131. package/src/application/models/colorsConfig.ts +28 -0
  132. package/src/application/models/conversationalSearchIds.ts +5 -0
  133. package/src/application/models/dataLayer.ts +45 -0
  134. package/src/application/models/domMutationContinuation.ts +7 -0
  135. package/src/application/models/domObservationStrategy.ts +9 -0
  136. package/src/application/models/events.ts +5 -0
  137. package/src/application/models/featureGates.ts +23 -0
  138. package/src/application/models/frontendConfig.ts +14 -0
  139. package/src/application/models/googleAnalyticsEvents.ts +8 -0
  140. package/src/application/models/graphql/index.ts +2 -0
  141. package/src/application/models/graphql/queries/getMerchantColorsQuery.ts +37 -0
  142. package/src/application/models/graphql/queries/getMerchantFrontendConfigQuery.ts +103 -0
  143. package/src/application/models/graphql/queries/getMerchantOrgIdQuery.ts +11 -0
  144. package/src/application/models/guards/api/index.ts +12 -0
  145. package/src/application/models/guards/api/isApiFormResponse.ts +90 -0
  146. package/src/application/models/guards/api/isApiFormSubmittedResponseAttributes.ts +37 -0
  147. package/src/application/models/guards/api/isApiOrderResponseAttributes.ts +155 -0
  148. package/src/application/models/guards/api/isApiOrgConfigResults.ts +277 -0
  149. package/src/application/models/guards/api/isApiOrganizationConfig.ts +207 -0
  150. package/src/application/models/guards/api/isApiPDPEventAttributes.ts +21 -0
  151. package/src/application/models/guards/api/isApiPLPEventAttributes.ts +41 -0
  152. package/src/application/models/guards/api/isApiPageResponseAttributes.ts +21 -0
  153. package/src/application/models/guards/api/isApiProductResponseAttributes.ts +85 -0
  154. package/src/application/models/guards/api/isApiProductSearchAttributes.ts +23 -0
  155. package/src/application/models/guards/api/isApiProductSearchFilterAttributes.ts +15 -0
  156. package/src/application/models/guards/api/isApiQueryTypedEventAttributes.ts +4 -0
  157. package/src/application/models/guards/api/isApiResponse.ts +39 -0
  158. package/src/application/models/guards/api/isApiReviewResponseAttributes.ts +30 -0
  159. package/src/application/models/guards/api/isApiReviewRichInformation.ts +37 -0
  160. package/src/application/models/guards/api/isApiSearchEventAttributes.ts +28 -0
  161. package/src/application/models/guards/api/isApiSuggestion.ts +36 -0
  162. package/src/application/models/guards/api/isApiSuggestionClickedEventAttributes.ts +9 -0
  163. package/src/application/models/guards/api/isApiTextResponseAttributes.ts +9 -0
  164. package/src/application/models/guards/api/isApiUserEvent.ts +25 -0
  165. package/src/application/models/guards/graphQL/isGraphQLColorsConfig.ts +50 -0
  166. package/src/application/models/guards/isBaseEcommerceEvent.ts +17 -0
  167. package/src/application/models/guards/isGA4EcommerceEvent.ts +17 -0
  168. package/src/application/models/guards/isLegacyUAEcommerceEvent.ts +17 -0
  169. package/src/application/models/guards/isMobilePLPChatPlacementParameter.ts +11 -0
  170. package/src/application/models/guards/isSpanxTakeAQuizCtaParameter.ts +4 -0
  171. package/src/application/models/guards/isVariantInfo.ts +37 -0
  172. package/src/application/models/guards/utils.ts +43 -0
  173. package/src/application/models/index.ts +20 -0
  174. package/src/application/models/localStorageEventListener.ts +4 -0
  175. package/src/application/models/message.ts +146 -0
  176. package/src/application/models/mobilePLPChatPlacementParameter.ts +3 -0
  177. package/src/application/models/orgsEnum.ts +36 -0
  178. package/src/application/models/productExperiment.ts +5 -0
  179. package/src/application/models/spanxTakeAQuizCtaParameter.ts +4 -0
  180. package/src/application/models/spiffyWidgets.ts +16 -0
  181. package/src/application/models/supportedOrgs.ts +137 -0
  182. package/src/application/models/utilityTypes/camelCase.ts +87 -0
  183. package/src/application/models/utilityTypes/camelCasedPropertiesDeep.ts +80 -0
  184. package/src/application/models/utilityTypes/delimiterCase.ts +121 -0
  185. package/src/application/models/utilityTypes/delimiterCasedPropertiesDeep.ts +98 -0
  186. package/src/application/models/utilityTypes/index.ts +1 -0
  187. package/src/application/models/utilityTypes/internal.ts +93 -0
  188. package/src/application/models/utilityTypes/primitive.ts +8 -0
  189. package/src/application/models/utilityTypes/snakeCasedPropertiesDeep.ts +49 -0
  190. package/src/application/models/utilityTypes/splitWords.ts +76 -0
  191. package/src/application/models/utilityTypes/trim.ts +28 -0
  192. package/src/application/models/utilityTypes/unknownArray.ts +25 -0
  193. package/src/application/models/utils/snakeToCamelTransformer.ts +90 -0
  194. package/src/application/models/utils/stringToFulfillmentDisplayStatusEnumValue.ts +68 -0
  195. package/src/application/models/validators/validateGraphQLColorsConfig.ts +29 -0
  196. package/src/application/models/validators/validateGraphQLFrontendConfig.ts +594 -0
  197. package/src/application/models/validators/validateGraphQLOrgId.ts +7 -0
  198. package/src/application/models/validators/validateMobilePLPChatPlacementParameter.ts +14 -0
  199. package/src/application/models/validators/validateOrgConfigResults.ts +47 -0
  200. package/src/application/models/validators/validateOrganizationConfig.ts +37 -0
  201. package/src/application/models/validators/validateResponse.ts +187 -0
  202. package/src/application/models/validators/validateSuggestion.ts +16 -0
  203. package/src/application/models/validators/validateUserEvent.ts +110 -0
  204. package/src/application/models/variantInfo/index.ts +1 -0
  205. package/src/application/models/variantInfo/pageVisitInfo.ts +6 -0
  206. package/src/application/models/variantInfo/plpInfo.ts +3 -0
  207. package/src/application/models/variantInfo/productInfo.ts +5 -0
  208. package/src/application/models/variantInfo/variantInfo.ts +23 -0
  209. package/src/application/service/cachingService.ts +84 -0
  210. package/src/application/service/cdnService.ts +18 -0
  211. package/src/application/service/customerService/index.ts +8 -0
  212. package/src/application/service/customerService/providers/UnsupportedCustomerService.ts +15 -0
  213. package/src/application/service/customerService/types.ts +31 -0
  214. package/src/application/service/domMutationObserver.ts +320 -0
  215. package/src/application/service/domMutations/GridInsertionService.ts +123 -0
  216. package/src/application/service/domMutations/dataLayer/dataLayerEventsListener.ts +99 -0
  217. package/src/application/service/domMutations/domInsertionService.ts +90 -0
  218. package/src/application/service/domMutations/domMutationListener.ts +15 -0
  219. package/src/application/service/domMutations/domMutationListenerState.ts +52 -0
  220. package/src/application/service/domMutations/floatingChat/embeddedChatsPlacementsListener.ts +41 -0
  221. package/src/application/service/domMutations/gladly/gladlyListener.ts +61 -0
  222. package/src/application/service/domMutations/spiffy/orgs/common/kustomerVisibilityListener.ts +41 -0
  223. package/src/application/service/domMutations/spiffy/orgs/common/orgsCommonDataLayerListener.ts +119 -0
  224. package/src/application/service/environmentService.ts +51 -0
  225. package/src/application/service/featureFlagService.ts +130 -0
  226. package/src/application/service/kustomerIntegrationService.ts +111 -0
  227. package/src/application/service/localStorageService.ts +77 -0
  228. package/src/application/service/pageVariantService.ts +779 -0
  229. package/src/application/service/searchService.ts +140 -0
  230. package/src/application/service/sessionStorageService.ts +27 -0
  231. package/src/application/service/shopifyUrlService.ts +63 -0
  232. package/src/application/service/userIdentityService.ts +114 -0
  233. package/src/application/service/windowChatToggleService.ts +71 -0
  234. package/src/application/service/windowDataLayerService.ts +181 -0
  235. package/src/application/service/windowFrontendConfigService.ts +104 -0
  236. package/src/application/utils/__tests__/divideArrays.test.ts +14 -0
  237. package/src/application/utils/analyticsUtils.ts +110 -0
  238. package/src/application/utils/coreContextToApiContext.ts +11 -0
  239. package/src/application/utils/coreUserEventToApiUserEvent.ts +106 -0
  240. package/src/application/utils/divideArray.ts +7 -0
  241. package/src/application/utils/domObserver.ts +96 -0
  242. package/src/application/utils/elementObserver.ts +246 -0
  243. package/src/application/utils/imageFilter.ts +12 -0
  244. package/src/application/utils/index.ts +3 -0
  245. package/src/application/utils/merchantUtils.ts +16 -0
  246. package/src/application/utils/messageFromFormSubmittedEvent.ts +31 -0
  247. package/src/application/utils/messageFromQueryEvent.ts +38 -0
  248. package/src/application/utils/messageFromResponse.ts +133 -0
  249. package/src/application/utils/messageFromSuggestionEvent.ts +32 -0
  250. package/src/application/utils/mouseEventTypes.ts +1 -0
  251. package/src/application/utils/mutationHelper.ts +51 -0
  252. package/src/application/utils/nextMessageRequestToApiRequest.ts +31 -0
  253. package/src/application/utils/nodeSelector.ts +133 -0
  254. package/src/application/utils/overrides.ts +196 -0
  255. package/src/application/utils/stringUtils.ts +55 -0
  256. package/src/application/utils/supportedEventRequestToApiRequest.ts +12 -0
  257. package/src/application/utils/urlsParser.ts +53 -0
  258. package/src/application/utils/validation.ts +5 -0
  259. package/src/atoms/app/index.ts +57 -0
  260. package/src/atoms/app/variant.ts +261 -0
  261. package/src/atoms/atomStore.ts +34 -0
  262. package/src/atoms/chat/chatState.ts +44 -0
  263. package/src/atoms/chat/form.ts +19 -0
  264. package/src/atoms/chat/index.ts +38 -0
  265. package/src/atoms/chat/lastMessage.ts +11 -0
  266. package/src/atoms/chat/messageQueue.ts +65 -0
  267. package/src/atoms/chat/performanceMetrics.ts +84 -0
  268. package/src/atoms/chat/renderedWidgetRefs.ts +28 -0
  269. package/src/atoms/chat/replies.ts +51 -0
  270. package/src/atoms/chat/suggestions.ts +36 -0
  271. package/src/atoms/globalSearch.ts +12 -0
  272. package/src/atoms/index.ts +5 -0
  273. package/src/atoms/org/customerService.ts +13 -0
  274. package/src/atoms/org/graphqlConfig.ts +27 -0
  275. package/src/atoms/org/index.ts +7 -0
  276. package/src/atoms/org/merchantCss.ts +44 -0
  277. package/src/atoms/org/org.ts +256 -0
  278. package/src/atoms/org/orgAnalyticsConfig.ts +28 -0
  279. package/src/atoms/org/orgPageConfig.ts +38 -0
  280. package/src/atoms/org/orgUIConfig.ts +122 -0
  281. package/src/atoms/search/chatSearch.ts +293 -0
  282. package/src/atoms/search/index.ts +2 -0
  283. package/src/atoms/search/productFilters.ts +207 -0
  284. package/src/atoms/search/productSorter.ts +23 -0
  285. package/src/atoms/search/searchAPI.ts +194 -0
  286. package/src/atoms/search/types.ts +55 -0
  287. package/src/atoms/search/utils.ts +18 -0
  288. package/src/config/divIds.ts +27 -0
  289. package/src/config/locators/components/chat/entrypoints.ts +13 -0
  290. package/src/config/locators/components/chat/index.ts +23 -0
  291. package/src/config/locators/components/chat/preview.ts +13 -0
  292. package/src/config/locators/components/chat/variants/index.ts +16 -0
  293. package/src/config/locators/components/common/buttons.ts +6 -0
  294. package/src/config/locators/components/common/cards.ts +18 -0
  295. package/src/config/locators/components/common/links.ts +1 -0
  296. package/src/config/locators/components/common/tables.ts +2 -0
  297. package/src/config/locators/components/floating-button.ts +2 -0
  298. package/src/config/locators/components/index.ts +3 -0
  299. package/src/config/locators/components/report-issue.ts +21 -0
  300. package/src/config/locators/components/search/index.ts +5 -0
  301. package/src/config/locators/components/shadow-dom.ts +1 -0
  302. package/src/config/locators/embedded.ts +21 -0
  303. package/src/config/locators/index.ts +3 -0
  304. package/src/config/socialProofClasses.ts +17 -0
  305. package/src/contexts/chatContext.tsx +451 -0
  306. package/src/contexts/enviveConfigContext.tsx +70 -0
  307. package/src/contexts/index.ts +4 -0
  308. package/src/contexts/systemSettingsContext.tsx +61 -0
  309. package/src/contexts/types.ts +1059 -0
  310. package/src/enabled-features.ts +83 -0
  311. package/src/events/event-types.ts +11 -0
  312. package/src/events/index.ts +52 -0
  313. package/src/events/registerAnalyticsListeners.ts +49 -0
  314. package/src/extension.ts +63 -0
  315. package/src/hooks/index.ts +22 -0
  316. package/src/hooks/useBlockBackButton.ts +29 -0
  317. package/src/hooks/useChatToggle.ts +66 -0
  318. package/src/hooks/useCustomerSupportHandoff.ts +39 -0
  319. package/src/hooks/useDebounce.ts +17 -0
  320. package/src/hooks/useDynamicVariants.ts +210 -0
  321. package/src/hooks/useElementObserver.ts +245 -0
  322. package/src/hooks/useFileUpload.ts +61 -0
  323. package/src/hooks/useGrabAndScroll.ts +133 -0
  324. package/src/hooks/useHideElements.ts +82 -0
  325. package/src/hooks/useHorizontalScrollAnimation.ts +115 -0
  326. package/src/hooks/useImageResolver.ts +51 -0
  327. package/src/hooks/useIntersection.ts +28 -0
  328. package/src/hooks/useIsSmallScreen.ts +23 -0
  329. package/src/hooks/useMessageFilter.ts +49 -0
  330. package/src/hooks/useMessageScrollObserver.ts +47 -0
  331. package/src/hooks/useReducedMotionWithOverride.ts +15 -0
  332. package/src/hooks/useSearch.tsx +433 -0
  333. package/src/hooks/useSnapCalculator.ts +38 -0
  334. package/src/hooks/useSnapControl.ts +155 -0
  335. package/src/hooks/useSystemSettingsContext.ts +12 -0
  336. package/src/hooks/useTrackComponentVisibleEvent.ts +52 -0
  337. package/src/hooks/useUpdateAnalyticsProps.ts +56 -0
  338. package/src/hooks/utils.ts +153 -0
  339. package/src/index.ts +31 -0
  340. package/src/initialize.ts +163 -0
  341. package/src/interceptors/types.ts +6 -0
  342. package/src/interceptors/useFormEscalation.ts +40 -0
  343. package/src/interceptors/useMessageInterceptor.ts +32 -0
  344. package/src/main.ts +85 -0
  345. package/src/main.tsx +123 -0
  346. package/src/merchants/bandolier/bandolier.ts +1389 -0
  347. package/src/merchants/carpe/carpe.ts +656 -0
  348. package/src/merchants/coterie/coterie.ts +280 -0
  349. package/src/merchants/default.ts +193 -0
  350. package/src/merchants/dreamlandBaby/dreamlandBaby.ts +375 -0
  351. package/src/merchants/fiveCbd/fiveCbd.ts +697 -0
  352. package/src/merchants/forLoveAndLemons/forLoveAndLemons.ts +721 -0
  353. package/src/merchants/greenpan/greenpan.ts +440 -0
  354. package/src/merchants/grooveLife/grooveLife.ts +386 -0
  355. package/src/merchants/homegrownCannabis/homegrownCannabis.ts +468 -0
  356. package/src/merchants/init-merchant.sh +53 -0
  357. package/src/merchants/jackArcher/jackArcher.ts +974 -0
  358. package/src/merchants/jordanCraig/jordanCraig.ts +1927 -0
  359. package/src/merchants/kindredBravely/kindredBravely.ts +529 -0
  360. package/src/merchants/kutFromTheKloth/kutFromTheKloth.ts +418 -0
  361. package/src/merchants/larryAndSerges/larryAndSerges.ts +314 -0
  362. package/src/merchants/leapsAndRebounds/leapsAndRebounds.ts +424 -0
  363. package/src/merchants/longevityrx/longevityrx.ts +368 -0
  364. package/src/merchants/lookOptic/lookOptic.ts +323 -0
  365. package/src/merchants/mantraBrand/mantraBrand.ts +838 -0
  366. package/src/merchants/medterra/medterra.ts +670 -0
  367. package/src/merchants/modells/modells.ts +546 -0
  368. package/src/merchants/pressedFloral/pressedFloral.ts +734 -0
  369. package/src/merchants/skinPerfection/skinPerfection.ts +379 -0
  370. package/src/merchants/snapSupplements/snapSupplements.ts +325 -0
  371. package/src/merchants/spanx/spanx.ts +810 -0
  372. package/src/merchants/spanx/spanxStaging.ts +942 -0
  373. package/src/merchants/supergoop/supergoop.ts +376 -0
  374. package/src/merchants/uniqueVintage/uniqueVintage.ts +1314 -0
  375. package/src/merchants/uniqueVintage/views/useUniqueVintageChatSearch.ts +147 -0
  376. package/src/merchants/venaCbd/venaCbd.ts +410 -0
  377. package/src/merchants/westonJonBoucher/westonJonBoucher.ts +473 -0
  378. package/src/merchants/wineEnthusiast/wineEnthusiast.ts +990 -0
  379. package/src/merchants/wolfMattress/wolfMattress.ts +411 -0
  380. package/src/merchants/wolfTactical/wolfTactical.ts +389 -0
  381. package/src/types/custservice-types.ts +28 -0
  382. package/src/types/search-filter-types.ts +111 -0
  383. package/src/types/suggestionBarV2-types.ts +4 -0
  384. package/src/types/test-types.ts +3 -0
  385. package/src/types.ts +66 -0
@@ -0,0 +1,320 @@
1
+ import { DomMutationObserverContinuation } from 'src/application/models/domMutationContinuation';
2
+ import { DomMutationListenerState } from './domMutations/domMutationListenerState';
3
+ import Logger from '../logging/logger';
4
+ import { DomMutationListener } from './domMutations/domMutationListener';
5
+ import { DomObservationStrategy } from '../models/domObservationStrategy';
6
+
7
+ /**
8
+ * - When used before page load/render, an unoptimized MutationObserver callback can significantly delay page load time,
9
+ * especially on complex pages.
10
+ * - The callback runs as a microtask and can block further DOM processing, often firing hundreds or even thousands of times
11
+ * per second on a complex page. Many libraries don't optimize for this, resulting in clean-looking but potentially slow code.
12
+ *
13
+ * Best Practices for MutationObserver Optimization:
14
+ * - **Profiling:** Use DevTools Profiler to ensure the observer callback uses less than 0.1% of total CPU during page load.
15
+ * - **Avoid Forced Layouts:** Refrain from triggering synchronous layouts by accessing properties like `offsetTop`.
16
+ * - **Prefer Native DOM:** Use native DOM methods instead of complex frameworks like jQuery to reduce overhead.
17
+ * - **Use `attributeFilter`:** When observing attributes, limit the observations to specific attributes, e.g., `attributeFilter: ['attr1', 'attr2']`.
18
+ * - **Non-recursive Observations:** Observe direct parent elements with `subtree: false` whenever possible. For instance, initially observe `document` recursively, disconnect on success, and reattach a non-recursive observer to the parent element.
19
+ * - **Direct Element Access:** When waiting for a single element with an `id`, use `getElementById` instead of looping over the `mutations` array.
20
+ * - **HTMLCollection Checks:** For rare elements (e.g., `<iframe>` or `<object>`), use live HTMLCollections from `getElementsByTagName` or `getElementsByClassName` rather than enumerating mutations, especially with large numbers of mutations.
21
+ * - **Avoid `querySelectorAll`:** Use `querySelector` first for an initial check; if successful, proceed with `querySelectorAll` only when absolutely necessary. This combo is often faster on average.
22
+ * - **Legacy Browser Compatibility:**
23
+ * - Avoid `Array` methods like `forEach`, `filter`, etc., and ES2015 loops (`for...of`) in pre-2018 Chrome for MutationObserver callbacks,
24
+ * as these can be up to 100x slower than traditional `for` loops.
25
+ * - Alternatively, use a fast functional library like lodash on older browsers. Chrome 2018+ inlines array methods efficiently.
26
+ * - **Scheduled Re-checking:** For visual updates, if you can determine elements are offscreen, disconnect the observer and use `setTimeout(fn, 0)` to delay processing until initial parsing/layout is complete. After that, use `requestAnimationFrame` for chunked processing.
27
+ * - **Debounce Long Processing:** If the callback processing is lengthy, causing unresponsiveness or jank, use debouncing by accumulating mutations in an array and scheduling processing with `setTimeout`, `requestIdleCallback`, or `requestAnimationFrame`.
28
+ *
29
+ * Example of Debounce Setup:
30
+ * ```typescript
31
+ * const queue: MutationRecord[] = [];
32
+ * const mo = new MutationObserver(mutations => {
33
+ * if (!queue.length) requestAnimationFrame(process);
34
+ * queue.push(...mutations);
35
+ * });
36
+ * function process() {
37
+ * for (const mutations of queue) {
38
+ * // Processing logic
39
+ * }
40
+ * queue.length = 0;
41
+ * }
42
+ * ```
43
+ * Note: `requestAnimationFrame` only triggers when the page is visible.
44
+ */
45
+ class DomMutationObserver {
46
+ private static observerVersion = 'v1.24';
47
+
48
+ private static individualObservers = false;
49
+
50
+ private static deferredExecutionDelay = 500;
51
+
52
+ private static startingPollingInterval = 250;
53
+
54
+ private static endingPollingInterval = 2000;
55
+
56
+ private static useDebounce = true;
57
+
58
+ private static listeners: DomMutationListenerState[] = [];
59
+
60
+ private static mutationsQueuePerListener: Map<DomMutationListenerState, MutationRecord[]> =
61
+ new Map();
62
+
63
+ private static mutationsQueue: MutationRecord[] = [];
64
+
65
+ private static observationStrategy: DomObservationStrategy = DomObservationStrategy.None;
66
+
67
+ static addDomMutationListener(listener: DomMutationListener): void {
68
+ // check if the listener is already added
69
+ const existingListener = this.listeners.find(
70
+ (l) => l.getListener().getName() === listener.getName(),
71
+ );
72
+ if (existingListener) {
73
+ // remove
74
+ Logger.logDebug(
75
+ `DomMutationObserver::addDomMutationListener removing existing listener ${existingListener.getListener().getName()}`,
76
+ );
77
+ this.disconnect(existingListener.getListener());
78
+ // remove it from the list
79
+ this.listeners = this.listeners.filter(
80
+ (l) => l.getListener().getName() !== listener.getName(),
81
+ );
82
+ }
83
+ const listenerState = new DomMutationListenerState(listener);
84
+ this.listeners.push(listenerState);
85
+ }
86
+
87
+ static invokeListener(l: DomMutationListenerState) {
88
+ if (l.getCurrentState() === DomMutationObserverContinuation.STOP) {
89
+ return;
90
+ }
91
+
92
+ const previousState = l.getCurrentState();
93
+ try {
94
+ const continuation = l.getListener().onDomChange();
95
+ l.setCurrentState(continuation);
96
+ l.incrementCurrentTryCount();
97
+ if (l.getListener().triesMax() > 0 && l.getCurrentTryCount() >= l.getListener().triesMax()) {
98
+ l.setCurrentState(DomMutationObserverContinuation.STOP);
99
+ Logger.logDebug(
100
+ `DomMutationObserver invokeListener ${l.getListener().getName()} tries exceeded! max: ${l.getListener().triesMax()} current: ${l.getCurrentTryCount()}`,
101
+ );
102
+ }
103
+ } catch (e) {
104
+ Logger.logError(
105
+ `DomMutationObserver invokeListener ${l?.getListener()?.getName()} state:${previousState}->${DomMutationObserverContinuation.STOP} error: ${e}`,
106
+ e,
107
+ );
108
+ l.setCurrentState(DomMutationObserverContinuation.STOP); // Un-trapped exceptions halt the observer
109
+ }
110
+ }
111
+
112
+ static observeOnDomReady(strategy: DomObservationStrategy) {
113
+ Logger.logDebug(`DomMutationObserver::observeOnDomReady ${document.readyState}`);
114
+ if (document.readyState === 'loading') {
115
+ document.addEventListener('DOMContentLoaded', () => {
116
+ this.observe(strategy);
117
+ });
118
+ } else {
119
+ this.observe(strategy);
120
+ }
121
+ }
122
+
123
+ static observe(strategyRequested: DomObservationStrategy) {
124
+ // We can only use the given polling strategy if no prior strategy has been set
125
+ // This is to prevent the listeners from being re-attached with a different strategy
126
+ // and have multiple strategies running at the same time!
127
+ if (this.observationStrategy === DomObservationStrategy.None) {
128
+ this.observationStrategy = strategyRequested;
129
+ }
130
+
131
+ const strategy = this.observationStrategy;
132
+ const listenersNotBound = this.listeners.filter((l) => !l.getIsAttached());
133
+ if (listenersNotBound.length === 0) {
134
+ Logger.logDebug(
135
+ `DomMutationObserver::observe (${this.observerVersion}) no listeners to attach (${listenersNotBound.length}/${this.listeners.length})`,
136
+ );
137
+ return;
138
+ }
139
+
140
+ Logger.logInfo(
141
+ `DomMutationObserver::observe (${this.observerVersion}) listeners to attach strategy=${strategy} (${listenersNotBound.length}/${this.listeners.length})`,
142
+ );
143
+
144
+ switch (strategy) {
145
+ case DomObservationStrategy.Polling:
146
+ this.observeByPolling(listenersNotBound);
147
+ break;
148
+ default:
149
+ if (this.useDebounce) {
150
+ if (this.individualObservers) {
151
+ this.observeIndividuallyWithDebounce(listenersNotBound);
152
+ } else {
153
+ this.observeCollectivelyWithDebounce(listenersNotBound);
154
+ }
155
+ } else {
156
+ if (this.individualObservers) {
157
+ this.observeIndividually(listenersNotBound);
158
+ } else {
159
+ this.observeCollectively(listenersNotBound);
160
+ }
161
+ }
162
+ break;
163
+ }
164
+
165
+ listenersNotBound.forEach((l) => {
166
+ l.setIsAttached(true);
167
+ });
168
+
169
+ DomMutationObserver.runOnceImmediatelyIfRequired(listenersNotBound);
170
+ }
171
+
172
+ private static runOnceImmediatelyIfRequired(listenersNotBound: DomMutationListenerState[]) {
173
+ listenersNotBound.forEach((l) => {
174
+ l.setIsAttached(true);
175
+ if (l.getListener().runOnceImmediately()) {
176
+ this.invokeListener(l);
177
+ }
178
+ });
179
+ }
180
+
181
+ private static observeIndividuallyWithDebounce(listenersNotBound: DomMutationListenerState[]) {
182
+ for (let i = 0; i < listenersNotBound.length; i++) {
183
+ // in some older browsers forEach is less performant
184
+ const l = listenersNotBound[i];
185
+ const observer = new MutationObserver((mutations) => {
186
+ // Avoid blocking the main thread and allow the browser to render, letting the callback run in the next microtask
187
+ let mutationQueueForListener = this.mutationsQueuePerListener.get(l);
188
+ if (!mutationQueueForListener) {
189
+ this.mutationsQueuePerListener.set(l, []);
190
+ mutationQueueForListener = this.mutationsQueuePerListener.get(l) || [];
191
+ }
192
+ // requestAnimationFrame or setTimeout can be used here
193
+ if (!mutationQueueForListener.length) {
194
+ setTimeout(() => {
195
+ this.invokeListener(l);
196
+ mutationQueueForListener.length = 0;
197
+ }, this.deferredExecutionDelay);
198
+ }
199
+ mutationQueueForListener.push(...mutations);
200
+ });
201
+ observer.observe(document.body, { childList: true, subtree: true });
202
+ }
203
+ }
204
+
205
+ private static observeIndividually(listenersNotBound: DomMutationListenerState[]) {
206
+ for (let i = 0; i < listenersNotBound.length; i++) {
207
+ // in some older browsers forEach is less performant
208
+ const l = listenersNotBound[i];
209
+ const observer = new MutationObserver(() => {
210
+ if (l.getCurrentState() === DomMutationObserverContinuation.CONTINUE) {
211
+ l.setCurrentState(DomMutationObserverContinuation.EXECUTING);
212
+ // Avoid blocking the main thread and allow the browser to render, letting the callback run in the next microtask
213
+ setTimeout(() => {
214
+ this.invokeListener(l);
215
+ }, this.deferredExecutionDelay);
216
+ }
217
+ });
218
+ observer.observe(document.body, { childList: true, subtree: true });
219
+ }
220
+ }
221
+
222
+ private static observeCollectivelyWithDebounce(listenersNotBound: DomMutationListenerState[]) {
223
+ const observer = new MutationObserver((mutations) => {
224
+ if (!this.mutationsQueue.length) {
225
+ setTimeout(() => {
226
+ for (let i = 0; i < listenersNotBound.length; i++) {
227
+ // in some older browsers forEach is less performant
228
+ const l = listenersNotBound[i];
229
+ if (l.getCurrentState() === DomMutationObserverContinuation.CONTINUE) {
230
+ l.setCurrentState(DomMutationObserverContinuation.EXECUTING);
231
+ this.invokeListener(l);
232
+ }
233
+ }
234
+ this.mutationsQueue.length = 0;
235
+ }, this.deferredExecutionDelay);
236
+ }
237
+ this.mutationsQueue.push(...mutations);
238
+ });
239
+ observer.observe(document.body, { childList: true, subtree: true });
240
+ }
241
+
242
+ private static observeCollectively(listenersNotBound: DomMutationListenerState[]) {
243
+ const observer = new MutationObserver(() => {
244
+ // Avoid blocking the main thread and allow the browser to render, letting the callback run in the next microtask
245
+ setTimeout(() => {
246
+ for (let i = 0; i < listenersNotBound.length; i++) {
247
+ // in some older browsers forEach is less performant
248
+ const l = listenersNotBound[i];
249
+ if (l.getCurrentState() === DomMutationObserverContinuation.CONTINUE) {
250
+ l.setCurrentState(DomMutationObserverContinuation.EXECUTING);
251
+ this.invokeListener(l);
252
+ }
253
+ }
254
+ }, this.deferredExecutionDelay);
255
+ });
256
+ observer.observe(document.body, { childList: true, subtree: true });
257
+ }
258
+
259
+ private static poll(
260
+ fn: () => Promise<void>,
261
+ startingInterval: number,
262
+ endingInterval: number,
263
+ ): void {
264
+ let intervalScalingStep = 1;
265
+ const executePoll = async () => {
266
+ try {
267
+ await fn();
268
+ } catch (error) {
269
+ Logger.logWarn('Polling error:', error);
270
+ } finally {
271
+ let interval = startingInterval * intervalScalingStep;
272
+ // take the min of the interval and the ending interval
273
+ interval = Math.min(interval, endingInterval);
274
+ if (interval < endingInterval) {
275
+ intervalScalingStep++;
276
+ }
277
+ setTimeout(executePoll, interval);
278
+ }
279
+ };
280
+ executePoll().then((_) => Logger.logDebug('Polling started'));
281
+ }
282
+
283
+ private static observeByPolling(listenersNotBound: DomMutationListenerState[]) {
284
+ this.poll(
285
+ async () => {
286
+ for (let i = 0; i < listenersNotBound.length; i++) {
287
+ // in some older browsers forEach is less performant
288
+ const l = listenersNotBound[i];
289
+ if (l.getCurrentState() === DomMutationObserverContinuation.CONTINUE) {
290
+ l.setCurrentState(DomMutationObserverContinuation.EXECUTING);
291
+ this.invokeListener(l);
292
+ }
293
+ }
294
+ },
295
+ this.startingPollingInterval,
296
+ this.endingPollingInterval,
297
+ );
298
+ }
299
+
300
+ static disconnect(listener: DomMutationListener) {
301
+ const listenerState = this.listeners.find((l) => l.getListener() === listener);
302
+
303
+ if (!listenerState) {
304
+ Logger.logDebug(`DomMutationObserver::disconnect listener not found`, undefined);
305
+ } else {
306
+ Logger.logDebug(`DomMutationObserver::disconnect listener found`, undefined);
307
+ listenerState.setCurrentState(DomMutationObserverContinuation.STOP);
308
+ }
309
+ }
310
+
311
+ static setStartingPollingInterval(interval: number) {
312
+ this.startingPollingInterval = interval;
313
+ }
314
+
315
+ static setEndingPollingInterval(interval: number) {
316
+ this.endingPollingInterval = interval;
317
+ }
318
+ }
319
+
320
+ export { DomMutationObserver };
@@ -0,0 +1,123 @@
1
+ import Logger from 'src/application/logging/logger';
2
+ import {
3
+ DOMInsertionService,
4
+ DOMInsertionType,
5
+ } from 'src/application/service/domMutations/domInsertionService';
6
+
7
+ enum GridInsertionType {
8
+ FIXED_1 = 'fixed-1',
9
+ FIXED_2 = 'fixed-2',
10
+ FIXED_3 = 'fixed-3',
11
+ FIXED_4 = 'fixed-4',
12
+ FIXED_5 = 'fixed-5',
13
+ FIXED_6 = 'fixed-6',
14
+ FIXED_7 = 'fixed-7',
15
+ FIXED_8 = 'fixed-8',
16
+ FIXED_9 = 'fixed-9',
17
+ FIXED_12 = 'fixed-12',
18
+ FIXED_13 = 'fixed-13',
19
+ ROW_1 = 'row-1',
20
+ ROW_2 = 'row-2',
21
+ ROW_3 = 'row-3',
22
+ ROW_4 = 'row-4',
23
+ }
24
+
25
+ class GridInsertionService {
26
+ static insert(
27
+ gridInsertionType: GridInsertionType,
28
+ targetElement: Element,
29
+ nodeToInsert: HTMLElement,
30
+ insertionType: DOMInsertionType,
31
+ ) {
32
+ switch (gridInsertionType) {
33
+ case GridInsertionType.FIXED_1:
34
+ GridInsertionService.fixedInsert(targetElement, 1, nodeToInsert, insertionType);
35
+ break;
36
+ case GridInsertionType.FIXED_2:
37
+ GridInsertionService.fixedInsert(targetElement, 2, nodeToInsert, insertionType);
38
+ break;
39
+ case GridInsertionType.FIXED_3:
40
+ GridInsertionService.fixedInsert(targetElement, 3, nodeToInsert, insertionType);
41
+ break;
42
+ case GridInsertionType.FIXED_4:
43
+ GridInsertionService.fixedInsert(targetElement, 4, nodeToInsert, insertionType);
44
+ break;
45
+ case GridInsertionType.FIXED_5:
46
+ GridInsertionService.fixedInsert(targetElement, 5, nodeToInsert, insertionType);
47
+ break;
48
+ case GridInsertionType.FIXED_6:
49
+ GridInsertionService.fixedInsert(targetElement, 6, nodeToInsert, insertionType);
50
+ break;
51
+ case GridInsertionType.FIXED_7:
52
+ GridInsertionService.fixedInsert(targetElement, 7, nodeToInsert, insertionType);
53
+ break;
54
+ case GridInsertionType.FIXED_8:
55
+ GridInsertionService.fixedInsert(targetElement, 8, nodeToInsert, insertionType);
56
+ break;
57
+ case GridInsertionType.FIXED_9:
58
+ GridInsertionService.fixedInsert(targetElement, 9, nodeToInsert, insertionType);
59
+ break;
60
+ case GridInsertionType.FIXED_12:
61
+ GridInsertionService.fixedInsert(targetElement, 12, nodeToInsert, insertionType);
62
+ break;
63
+ case GridInsertionType.FIXED_13:
64
+ GridInsertionService.fixedInsert(targetElement, 13, nodeToInsert, insertionType);
65
+ break;
66
+ case GridInsertionType.ROW_1:
67
+ GridInsertionService.rowInsert(targetElement, nodeToInsert, 1);
68
+ break;
69
+ case GridInsertionType.ROW_2:
70
+ GridInsertionService.rowInsert(targetElement, nodeToInsert, 2);
71
+ break;
72
+ case GridInsertionType.ROW_3:
73
+ GridInsertionService.rowInsert(targetElement, nodeToInsert, 3);
74
+ break;
75
+ case GridInsertionType.ROW_4:
76
+ GridInsertionService.rowInsert(targetElement, nodeToInsert, 4);
77
+ break;
78
+ default:
79
+ Logger.logError(
80
+ `Skipping grid insertion: invalid grid insertion type "${gridInsertionType}". \
81
+ Valid types are: ${Object.keys(GridInsertionType).join(', ')}`,
82
+ undefined,
83
+ );
84
+ }
85
+ }
86
+
87
+ /**
88
+ * Inserts a node into a grid row at the specified index starting from index 1. If the index
89
+ * is higher than the total number of rows, the node will be inserted at the end of the grid.
90
+ * This function does not make any assumptions about the column span of the inserted node - that
91
+ * should be handled by the orgConfig instead.
92
+ *
93
+ * @param targetElement - The target element to insert the node into.
94
+ * @param nodeToInsert - The node to insert into the grid.
95
+ * @param rowIndex - The index of the row to insert the node into.
96
+ */
97
+ static rowInsert(targetElement: Element, nodeToInsert: HTMLElement, rowIndex: number) {
98
+ const node = nodeToInsert;
99
+ node.style.gridRow = rowIndex.toString();
100
+ node.style.gridColumn = '1/-1';
101
+ DOMInsertionService.insert(targetElement, node, DOMInsertionType.APPEND_CHILD);
102
+ }
103
+
104
+ static fixedInsert(
105
+ targetElement: Element,
106
+ insertionGridIndex: number,
107
+ nodeToInsert: HTMLElement,
108
+ insertionType: DOMInsertionType,
109
+ ) {
110
+ const childCount = targetElement.children.length;
111
+ const actualIndex = Math.min(insertionGridIndex - 1, childCount - 1);
112
+ const targetChild = targetElement.children[actualIndex];
113
+
114
+ if (!targetChild) {
115
+ Logger.logWarn('Skipping grid insertion: grid has no elements', undefined);
116
+ return;
117
+ }
118
+
119
+ DOMInsertionService.insert(targetChild, nodeToInsert, insertionType);
120
+ }
121
+ }
122
+
123
+ export { GridInsertionType, GridInsertionService };
@@ -0,0 +1,99 @@
1
+ import { OrgAnalyticsGoogleAnalyticsConfig } from 'src/application/models/api/orgAnalyticsConfig';
2
+ import { DomMutationListener } from '../domMutationListener';
3
+ import { DomMutationObserverContinuation } from '../../../models/domMutationContinuation';
4
+ import Logger from '../../../logging/logger';
5
+ import { WindowDataLayerService } from '../../windowDataLayerService';
6
+ import { StringUtils } from '../../../utils/stringUtils';
7
+
8
+ class DataLayerEventsListener implements DomMutationListener {
9
+ private readonly onDataLayerPush: (event: object) => void;
10
+
11
+ private readonly eventsToIntercept: string[] = [];
12
+
13
+ private readonly windowDataLayerService: WindowDataLayerService;
14
+
15
+ constructor(
16
+ onDataLayerPush: (event: object) => void,
17
+ eventsToIntercept: string[],
18
+ gaConfig: OrgAnalyticsGoogleAnalyticsConfig,
19
+ ) {
20
+ Logger.logDebug('[spiffy-ai] DataLayerEventsListener created', undefined);
21
+ this.onDataLayerPush = onDataLayerPush;
22
+ this.eventsToIntercept = eventsToIntercept;
23
+ this.windowDataLayerService = WindowDataLayerService.getSingletonInstanceOf(gaConfig);
24
+ }
25
+
26
+ runOnceImmediately(): boolean {
27
+ return true;
28
+ }
29
+
30
+ triesMax(): number {
31
+ return 10;
32
+ }
33
+
34
+ onDomChange(): DomMutationObserverContinuation {
35
+ if (!WindowDataLayerService.isDataLayerAvailable()) {
36
+ return DomMutationObserverContinuation.CONTINUE;
37
+ }
38
+
39
+ Logger.logDebug(`[spiffy-ai] ${this.getName()} [subscribing] to DataLayer push events`);
40
+ // eslint-disable-next-line @typescript-eslint/no-this-alias
41
+ const This = this;
42
+
43
+ function dataLayerPushProxyMethod() {
44
+ try {
45
+ // flatten the event if it is an array
46
+ const allArguments = WindowDataLayerService.tryFlattenArguments(arguments);
47
+ let event = allArguments[0];
48
+ if (!event) {
49
+ Logger.logDebug(
50
+ `[spiffy-ai] ${This.getName()} dataLayer event undefined. all_arguments=`,
51
+ allArguments,
52
+ );
53
+ return;
54
+ }
55
+
56
+ if (event?.length > 1) {
57
+ if (event[0] === 'event') {
58
+ const eventName = StringUtils.trimToNull(event[1]) || '';
59
+ if (This.eventsToIntercept.includes(eventName)) {
60
+ Logger.logDebug(
61
+ `[spiffy-ai] ${This.getName()} [datalayer-event] (1) event_name=${eventName} events_to_intercept=${This.eventsToIntercept} all_arguments=`,
62
+ allArguments,
63
+ );
64
+ event[2]['event'] = eventName;
65
+ This.onDataLayerPush(event[2]);
66
+ return;
67
+ }
68
+ }
69
+ }
70
+
71
+ let eventName = StringUtils.trimToNull(event['event']) || '';
72
+ if (!eventName && event['0']) {
73
+ // Handle this case: [Arguments] { '0': { event: 'event1' } }
74
+ eventName = StringUtils.trimToNull(event['0']['event']) || '';
75
+ event = event['0'];
76
+ }
77
+
78
+ if (This.eventsToIntercept.includes(eventName)) {
79
+ Logger.logDebug(
80
+ `[spiffy-ai] ${This.getName()} [datalayer-event] (2) event_name=${eventName} events_to_intercept=${This.eventsToIntercept} all_arguments=`,
81
+ arguments,
82
+ );
83
+ This.onDataLayerPush(event);
84
+ }
85
+ } catch (e) {
86
+ Logger.logError(`[spiffy-ai] ${This.getName()} Error in dataLayer.push: `, e);
87
+ }
88
+ }
89
+
90
+ this.windowDataLayerService.wrapWithProxyMethod('push', dataLayerPushProxyMethod);
91
+ return DomMutationObserverContinuation.STOP;
92
+ }
93
+
94
+ getName(): string {
95
+ return 'DataLayerEventsListener';
96
+ }
97
+ }
98
+
99
+ export { DataLayerEventsListener };
@@ -0,0 +1,90 @@
1
+ enum DOMInsertionType {
2
+ AFTER = 'after',
3
+ BEFORE = 'before',
4
+ ADJACENT_AFTER_BEGIN = 'adjacent-after-begin',
5
+ ADJACENT_BEFORE_BEGIN = 'adjacent-before-begin',
6
+ ADJACENT_AFTER_END = 'adjacent-after-end',
7
+ ADJACENT_BEFORE_END = 'adjacent-before-end',
8
+ APPEND_CHILD = 'append-child',
9
+ NONE = 'none',
10
+ }
11
+
12
+ export const DOMInsertionTypeToInsertPosition = (type: DOMInsertionType): InsertPosition => {
13
+ switch (type) {
14
+ case DOMInsertionType.ADJACENT_AFTER_BEGIN:
15
+ return 'afterbegin';
16
+ case DOMInsertionType.BEFORE:
17
+ case DOMInsertionType.ADJACENT_BEFORE_BEGIN:
18
+ return 'beforebegin';
19
+ case DOMInsertionType.ADJACENT_BEFORE_END:
20
+ return 'beforeend';
21
+ case DOMInsertionType.AFTER:
22
+ case DOMInsertionType.ADJACENT_AFTER_END:
23
+ default:
24
+ return 'afterend';
25
+ }
26
+ };
27
+
28
+ class DOMInsertionService {
29
+ static insert(
30
+ targetElement: Element,
31
+ nodeToInsert: HTMLElement,
32
+ insertionType: DOMInsertionType,
33
+ ) {
34
+ switch (insertionType) {
35
+ case DOMInsertionType.AFTER:
36
+ targetElement.after(nodeToInsert);
37
+ break;
38
+ case DOMInsertionType.BEFORE:
39
+ targetElement.before(nodeToInsert);
40
+ break;
41
+ case DOMInsertionType.ADJACENT_AFTER_BEGIN:
42
+ targetElement.insertAdjacentElement('afterbegin', nodeToInsert);
43
+ break;
44
+ case DOMInsertionType.ADJACENT_BEFORE_BEGIN:
45
+ targetElement.insertAdjacentElement('beforebegin', nodeToInsert);
46
+ break;
47
+ case DOMInsertionType.ADJACENT_AFTER_END:
48
+ targetElement.insertAdjacentElement('afterend', nodeToInsert);
49
+ break;
50
+ case DOMInsertionType.ADJACENT_BEFORE_END:
51
+ targetElement.insertAdjacentElement('beforeend', nodeToInsert);
52
+ break;
53
+ case DOMInsertionType.APPEND_CHILD:
54
+ targetElement.appendChild(nodeToInsert);
55
+ break;
56
+ case DOMInsertionType.NONE:
57
+ // No insertion
58
+ break;
59
+ // This should only happen if a new insertion type is added and not handled in the switch statement
60
+ default:
61
+ throw new Error(`Invalid insertion type: ${insertionType}`);
62
+ }
63
+ }
64
+
65
+ /**
66
+ * Get the insertion point for the widget.
67
+ * @param selector - The selector to find the insertion point.
68
+ * @param selectionIndex - For multiple matches, pick the nth element. Can be negative to insert from the end (-1 is the last element). (optional, default first)
69
+ * @returns The insertion point element or null if not found.
70
+ */
71
+ static getInsertionPoint(selector: string, selectionIndex?: number): Element | null {
72
+ let elem: Element | null = null;
73
+ if (selectionIndex === undefined || selectionIndex === 0) {
74
+ elem = document.querySelector(selector);
75
+ if (elem?.parentElement) {
76
+ return elem;
77
+ }
78
+ } else {
79
+ const elems = document.querySelectorAll(selector);
80
+ const minLength = selectionIndex > 0 ? selectionIndex + 1 : -selectionIndex;
81
+ const index = selectionIndex > 0 ? selectionIndex : elems.length + selectionIndex;
82
+ if (elems.length >= minLength && elems[index].parentElement) {
83
+ return elems[index];
84
+ }
85
+ }
86
+ return null;
87
+ }
88
+ }
89
+
90
+ export { DOMInsertionService, DOMInsertionType };
@@ -0,0 +1,15 @@
1
+ import { DomMutationObserverContinuation } from '../../models/domMutationContinuation';
2
+
3
+ interface DomMutationListener {
4
+ getName(): string;
5
+
6
+ runOnceImmediately(): boolean;
7
+
8
+ disconnect?(): void;
9
+
10
+ triesMax(): number;
11
+
12
+ onDomChange(): DomMutationObserverContinuation;
13
+ }
14
+
15
+ export type { DomMutationListener };