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

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 (83) hide show
  1. package/dist/{TrackComponentVisibleEvent-CXhKOwKQ.js → TrackComponentVisibleEvent-qNE0dIL1.js} +2 -2
  2. package/dist/{TrackComponentVisibleEvent-CgxCqrIt.cjs → TrackComponentVisibleEvent-tDRnrJi3.cjs} +2 -2
  3. package/dist/amplitudeContext-C6LHpqP_.cjs +290 -0
  4. package/dist/amplitudeContext-rEjGf57q.js +268 -0
  5. package/dist/application/utils/index.cjs +2 -2
  6. package/dist/application/utils/index.js +2 -2
  7. package/dist/atoms/app/index.d.cts +7 -7
  8. package/dist/atoms/app/index.d.ts +1 -1
  9. package/dist/atoms/chat/index.cjs +3 -3
  10. package/dist/atoms/chat/index.d.cts +26 -26
  11. package/dist/atoms/chat/index.d.ts +1 -1
  12. package/dist/atoms/chat/index.js +3 -3
  13. package/dist/atoms/globalSearch/index.d.cts +6 -6
  14. package/dist/atoms/globalSearch/index.d.ts +5 -5
  15. package/dist/atoms/org/index.d.cts +16 -16
  16. package/dist/atoms/org/index.d.ts +16 -16
  17. package/dist/atoms/search/index.cjs +5 -5
  18. package/dist/atoms/search/index.js +5 -5
  19. package/dist/atoms/search/utils.d.ts +1 -1
  20. package/dist/{cdnContext-CtrIlAqX.js → cdnContext-B8zWuGGT.js} +1 -1
  21. package/dist/{cdnContext-CaDyQ_5p.cjs → cdnContext-DzifgoNo.cjs} +1 -1
  22. package/dist/{chat-BjhQCyW_.js → chat-DChvXHjz.js} +3 -3
  23. package/dist/{chat-BkPax29G.cjs → chat-L_N0qaqs.cjs} +3 -3
  24. package/dist/{chatSearch-C3N3iIxu.cjs → chatSearch-B5whqPLW.cjs} +3 -3
  25. package/dist/{chatSearch-BsYlFvpv.js → chatSearch-so-qeiEL.js} +3 -3
  26. package/dist/{chatState-CJ52Ag_7.cjs → chatState-D9_aA1_h.cjs} +2 -2
  27. package/dist/{chatState-CXA1vF16.js → chatState-Dl5lyuKC.js} +2 -2
  28. package/dist/{commerce-api-rgj30eEp.js → commerce-api-DEFd5HUH.js} +2 -2
  29. package/dist/{commerce-api-DA1QGGMK.cjs → commerce-api-DqVD4NH8.cjs} +2 -2
  30. package/dist/contexts/amplitudeContext/index.cjs +2 -2
  31. package/dist/contexts/amplitudeContext/index.js +2 -2
  32. package/dist/contexts/cdnContext/index.cjs +1 -1
  33. package/dist/contexts/cdnContext/index.js +1 -1
  34. package/dist/contexts/chatContext/index.cjs +7 -7
  35. package/dist/contexts/chatContext/index.d.ts +4 -4
  36. package/dist/contexts/chatContext/index.js +7 -7
  37. package/dist/contexts/systemSettingsContext/index.d.cts +2 -2
  38. package/dist/contexts/systemSettingsContext/index.d.ts +4 -4
  39. package/dist/contexts/userIdentityContext/index.cjs +4 -4
  40. package/dist/contexts/userIdentityContext/index.js +4 -4
  41. package/dist/frontendConfig-Cawh5iqv.d.ts +1 -1
  42. package/dist/frontendConfig-iZipB5FG.d.cts +1 -1
  43. package/dist/hooks/AmplitudeOperations/index.cjs +3 -3
  44. package/dist/hooks/AmplitudeOperations/index.js +3 -3
  45. package/dist/hooks/CdnOperations/index.cjs +1 -1
  46. package/dist/hooks/CdnOperations/index.js +1 -1
  47. package/dist/hooks/ChatToggle/index.cjs +3 -3
  48. package/dist/hooks/ChatToggle/index.js +3 -3
  49. package/dist/hooks/ChatToggleAnalytics/index.cjs +4 -4
  50. package/dist/hooks/ChatToggleAnalytics/index.js +4 -4
  51. package/dist/hooks/GrabAndScroll/index.d.cts +2 -2
  52. package/dist/hooks/IdentifyUser/index.cjs +4 -4
  53. package/dist/hooks/IdentifyUser/index.js +4 -4
  54. package/dist/hooks/Search/index.cjs +19 -1157
  55. package/dist/hooks/Search/index.js +17 -1156
  56. package/dist/hooks/SystemSettingsContext/index.d.ts +2 -2
  57. package/dist/hooks/TrackComponentVisibleEvent/index.cjs +2 -2
  58. package/dist/hooks/TrackComponentVisibleEvent/index.js +2 -2
  59. package/dist/hooks/UpdateAnalyticsProps/index.cjs +1 -1
  60. package/dist/hooks/UpdateAnalyticsProps/index.js +1 -1
  61. package/dist/{search-Csh2n66W.cjs → search-CneZkauD.cjs} +2 -2
  62. package/dist/{search-DkiqkogN.js → search-DbZeJYdg.js} +2 -2
  63. package/dist/{useAmplitudeOperations-zIRSqmMW.js → useAmplitudeOperations-B9HjVe6G.js} +2 -2
  64. package/dist/{useAmplitudeOperations-Bo6YNbTV.cjs → useAmplitudeOperations-BdvxAzuI.cjs} +2 -2
  65. package/dist/userIdentityContext-CPztaX6-.js +126 -0
  66. package/dist/userIdentityContext-DdySlQBz.cjs +143 -0
  67. package/dist/{utils-C1ErYSoW.js → utils-B4PvsKVY.js} +2 -2
  68. package/dist/{utils-mqfncrhI.cjs → utils-BRVP7Mnu.cjs} +2 -2
  69. package/dist/utils-CBD4g2Nc.d.cts +1 -1
  70. package/dist/utils-QKFAbPT6.d.ts +1 -1
  71. package/package.json +3 -2
  72. package/src/contexts/amplitudeContext/__tests__/amplitudeContext.test.tsx +525 -0
  73. package/src/contexts/amplitudeContext/amplitudeContext.tsx +6 -2
  74. package/src/contexts/userIdentityContext/__tests__/userIdentityContext.test.tsx +0 -162
  75. package/src/contexts/userIdentityContext/userIdentityContext.tsx +13 -4
  76. package/dist/amplitudeContext-C8tT74Mi.cjs +0 -286
  77. package/dist/amplitudeContext-DCk6Va-j.js +0 -264
  78. package/dist/userIdentityContext-BqbNu7xu.cjs +0 -132
  79. package/dist/userIdentityContext-BxFH9FNQ.js +0 -115
  80. /package/dist/{AmplitudeOperations-ChZWcSsc.js → AmplitudeOperations-C-ieCm9m.js} +0 -0
  81. /package/dist/{AmplitudeOperations-JggIc1zD.cjs → AmplitudeOperations-p7APchq9.cjs} +0 -0
  82. /package/dist/{amplitudeContext-BItT9HmT.js → amplitudeContext-DbicJUzl.js} +0 -0
  83. /package/dist/{amplitudeContext-DPtyVv3Q.cjs → amplitudeContext-q3mggFSE.cjs} +0 -0
@@ -496,168 +496,6 @@ describe("UserIdentityProvider", () => {
496
496
  { timeout: 3000 }
497
497
  );
498
498
  });
499
-
500
- it("should not call identifyUser when context is not ready", async () => {
501
- const identifySpy = vi.spyOn(CommerceApiClient, "identifyUser");
502
- const logWarnSpy = vi.spyOn(Logger, "logWarn");
503
-
504
- // Mock localStorage to be unavailable
505
- const originalLocalStorage = window.localStorage;
506
- Object.defineProperty(window, "localStorage", {
507
- value: undefined,
508
- writable: true,
509
- });
510
-
511
- renderWithProviders(<IdentifyUserComponent />);
512
-
513
- await waitFor(() => {
514
- expect(screen.getByTestId("identify-button")).toBeInTheDocument();
515
- });
516
-
517
- await act(async () => {
518
- screen.getByTestId("identify-button").click();
519
- });
520
-
521
- await waitFor(() => {
522
- expect(logWarnSpy).toHaveBeenCalledWith(
523
- "[UserIdentityContext] Context not ready, skipping identifyUser",
524
- undefined
525
- );
526
- expect(identifySpy).not.toHaveBeenCalled();
527
- });
528
-
529
- // Restore localStorage
530
- Object.defineProperty(window, "localStorage", {
531
- value: originalLocalStorage,
532
- writable: true,
533
- });
534
- });
535
-
536
- it("should handle errors when identifyUser fails", async () => {
537
- const identifySpy = vi
538
- .spyOn(CommerceApiClient, "identifyUser")
539
- .mockRejectedValueOnce(new Error("API Error"));
540
- const logErrorSpy = vi.spyOn(Logger, "logError");
541
-
542
- renderWithProviders(<IdentifyUserComponent />);
543
-
544
- await waitFor(() => {
545
- expect(screen.getByTestId("identify-button")).toBeInTheDocument();
546
- });
547
-
548
- await act(async () => {
549
- screen.getByTestId("identify-button").click();
550
- });
551
-
552
- await waitFor(() => {
553
- expect(identifySpy).toHaveBeenCalled();
554
- expect(logErrorSpy).toHaveBeenCalledWith(
555
- "[spiffy-ai] Error identifying user",
556
- expect.any(Error)
557
- );
558
- });
559
- });
560
-
561
- it("should skip identifyUser when cdpUserId is not available", async () => {
562
- const identifySpy = vi.spyOn(CommerceApiClient, "identifyUser");
563
- const logWarnSpy = vi.spyOn(Logger, "logWarn");
564
-
565
- // The current implementation uses "UNKNOWN_CDP_USER_ID" as placeholder
566
- // and checks if cdpUserId is falsy, so it will skip
567
- renderWithProviders(<IdentifyUserComponent />);
568
-
569
- await waitFor(() => {
570
- expect(screen.getByTestId("identify-button")).toBeInTheDocument();
571
- });
572
-
573
- await act(async () => {
574
- screen.getByTestId("identify-button").click();
575
- });
576
-
577
- // Note: The current implementation checks `if (!cdpUserId)` but uses "UNKNOWN_CDP_USER_ID"
578
- // which is truthy, so this test may need adjustment based on actual behavior
579
- // For now, we'll verify the call was made (since "UNKNOWN_CDP_USER_ID" is truthy)
580
- await waitFor(() => {
581
- // The current code will call identifyUser because "UNKNOWN_CDP_USER_ID" is truthy
582
- // If the behavior changes, this test should be updated
583
- expect(identifySpy).toHaveBeenCalled();
584
- });
585
- });
586
- });
587
-
588
- describe("isReady state", () => {
589
- it("should be false when localStorage is not available", async () => {
590
- const originalLocalStorage = window.localStorage;
591
- Object.defineProperty(window, "localStorage", {
592
- value: undefined,
593
- writable: true,
594
- });
595
-
596
- renderWithProviders(<MockComponent />);
597
-
598
- await waitFor(() => {
599
- expect(screen.getByTestId("is-ready").textContent).toBe("false");
600
- });
601
-
602
- // Restore localStorage
603
- Object.defineProperty(window, "localStorage", {
604
- value: originalLocalStorage,
605
- writable: true,
606
- });
607
- });
608
-
609
- it("should be true when localStorage is available", async () => {
610
- renderWithProviders(<MockComponent />);
611
-
612
- await waitFor(() => {
613
- expect(screen.getByTestId("is-ready").textContent).toBe("true");
614
- });
615
- });
616
-
617
- it("should update when localStorage availability changes", async () => {
618
- renderWithProviders(<MockComponent />);
619
-
620
- await waitFor(() => {
621
- expect(screen.getByTestId("is-ready").textContent).toBe("true");
622
- });
623
-
624
- // Note: Testing localStorage becoming unavailable requires unmounting and remounting
625
- // the provider, which is complex. This test verifies the ready state is correct
626
- // when localStorage is available. The unavailable case is tested in the other test.
627
- });
628
- });
629
-
630
- describe("Context Value Memoization", () => {
631
- it("should maintain stable context value references", async () => {
632
- let contextValue1: UserIdentityContextType | undefined;
633
- let contextValue2: UserIdentityContextType | undefined;
634
-
635
- const Component1: React.FC = () => {
636
- const context = useUserIdentity();
637
- contextValue1 = context;
638
- return <div>Component1</div>;
639
- };
640
-
641
- const Component2: React.FC = () => {
642
- const context = useUserIdentity();
643
- contextValue2 = context;
644
- return <div>Component2</div>;
645
- };
646
-
647
- renderWithProviders(
648
- <>
649
- <Component1 />
650
- <Component2 />
651
- </>
652
- );
653
-
654
- await waitFor(() => {
655
- expect(contextValue1).toBeDefined();
656
- expect(contextValue2).toBeDefined();
657
- // Both components should receive the same context instance
658
- expect(contextValue1).toBe(contextValue2);
659
- });
660
- });
661
499
  });
662
500
  });
663
501
 
@@ -50,6 +50,8 @@ const UserIdentityContext = createContext<UserIdentityContextType | undefined>(
50
50
  export const UserIdentityProvider: React.FC<{ children: React.ReactNode }> = ({
51
51
  children,
52
52
  }) => {
53
+ const [localUserId, setLocalUserId] = useState<string | undefined>(undefined);
54
+ const setUserId = useSetAtom(userIdAtom);
53
55
  const {
54
56
  getItem,
55
57
  setItem,
@@ -180,7 +182,18 @@ export const UserIdentityProvider: React.FC<{ children: React.ReactNode }> = ({
180
182
  isReady,
181
183
  ]
182
184
  );
185
+ useEffect(() => {
186
+ if (isReady && !localUserId) {
187
+ const newUserId = getUserIdOrDefault();
188
+ console.log("useUserIdentity useEffect - setting localUserId", newUserId);
189
+ setLocalUserId(newUserId);
190
+ setUserId(newUserId);
191
+ }
192
+ }, [isReady, localUserId, setUserId]);
183
193
 
194
+ if (!localUserId) {
195
+ return null;
196
+ }
184
197
  return (
185
198
  <UserIdentityContext.Provider value={value}>
186
199
  {children}
@@ -190,10 +203,6 @@ export const UserIdentityProvider: React.FC<{ children: React.ReactNode }> = ({
190
203
 
191
204
  export const useUserIdentity = () => {
192
205
  const context = useContext(UserIdentityContext);
193
- const setUserId = useSetAtom(userIdAtom);
194
- useEffect(() => {
195
- setUserId(context?.getUserIdOrDefault() ?? "");
196
- }, [context, setUserId]);
197
206
  if (!context) {
198
207
  throw new Error(
199
208
  "useUserIdentity must be used within a UserIdentityProvider"
@@ -1,286 +0,0 @@
1
- const require_chunk = require('./chunk-CUT6urMc.cjs');
2
- const require_events = require('./events-DYY4l817.cjs');
3
- const require_models = require('./models-DqdLOi2I.cjs');
4
- const require_featureGates = require('./featureGates-qU_ulhpC.cjs');
5
- const require_logger = require('./logger-TBIl4uIH.cjs');
6
- const require_localStorageContext = require('./localStorageContext-NRP-CdmF.cjs');
7
- const require_enviveConfig = require('./enviveConfig-Dv9-esGV.cjs');
8
- const require_orgAnalyticsConfig = require('./orgAnalyticsConfig-Bm23fw4s.cjs');
9
- const require_app = require('./app-BbPSHefQ.cjs');
10
- const require_enviveConfigContext = require('./enviveConfigContext-D2OELZDR.cjs');
11
- const require_featureFlagServiceContext = require('./featureFlagServiceContext-CJyYItqu.cjs');
12
- let react = require("react");
13
- react = require_chunk.__toESM(react);
14
- let jotai = require("jotai");
15
- jotai = require_chunk.__toESM(jotai);
16
- let __amplitude_analytics_browser = require("@amplitude/analytics-browser");
17
- __amplitude_analytics_browser = require_chunk.__toESM(__amplitude_analytics_browser);
18
- let react_jsx_runtime = require("react/jsx-runtime");
19
- react_jsx_runtime = require_chunk.__toESM(react_jsx_runtime);
20
-
21
- //#region src/contexts/amplitudeContext/amplitudeContext.tsx
22
- let SpiffyMetricsEventName = /* @__PURE__ */ function(SpiffyMetricsEventName$1) {
23
- SpiffyMetricsEventName$1["BundleLoaded"] = "Bundle Loaded";
24
- SpiffyMetricsEventName$1["ChatLiveAgentBtnClick"] = "Chat Live Agent Btn Click";
25
- SpiffyMetricsEventName$1["ChatFloatingButtonVisible"] = "Chat Floating Button Visible";
26
- SpiffyMetricsEventName$1["ChatComponentVisible"] = "Chat Component Visible";
27
- SpiffyMetricsEventName$1["ChatComponentExpanded"] = "Chat Component Expanded";
28
- SpiffyMetricsEventName$1["ChatComponentCollapsed"] = "Chat Component Collapsed";
29
- SpiffyMetricsEventName$1["ChatUserMessageInput"] = "Chat User Message Input";
30
- SpiffyMetricsEventName$1["ChatSuggestionClicked"] = "Chat Suggestion Clicked";
31
- SpiffyMetricsEventName$1["ChatAssistantResponse"] = "Chat Assistant Response";
32
- SpiffyMetricsEventName$1["ProductCardClicked"] = "Product Card Clicked";
33
- SpiffyMetricsEventName$1["ProductReviewCardClicked"] = "Product Review Card Clicked";
34
- SpiffyMetricsEventName$1["AddToCartClicked"] = "Add to Cart Clicked";
35
- SpiffyMetricsEventName$1["PromptCardClicked"] = "Prompt Card Clicked";
36
- SpiffyMetricsEventName$1["SupportedEvent"] = "Supported Event";
37
- SpiffyMetricsEventName$1["SearchBackToResponseClicked"] = "Search Back to Response Clicked";
38
- SpiffyMetricsEventName$1["PerformanceMetrics"] = "Performance Metrics";
39
- SpiffyMetricsEventName$1["SearchBarClicked"] = "Search Bar Clicked";
40
- SpiffyMetricsEventName$1["OrderLookupStarted"] = "Order Lookup Started";
41
- SpiffyMetricsEventName$1["OrderLookupFormSubmitted"] = "Order Lookup Form Submitted";
42
- SpiffyMetricsEventName$1["SearchComponentVisible"] = "Search Component Visible";
43
- SpiffyMetricsEventName$1["SearchZeroStateSuggestionClicked"] = "Search Zero State Suggestion Clicked";
44
- SpiffyMetricsEventName$1["SearchInputStarted"] = "Search Input Started";
45
- SpiffyMetricsEventName$1["SearchQuerySubmitted"] = "Search Query Submitted";
46
- SpiffyMetricsEventName$1["SearchResultsViewed"] = "Search Results Viewed";
47
- SpiffyMetricsEventName$1["SearchTimeToFirstClick"] = "Search Time to First Click";
48
- SpiffyMetricsEventName$1["SearchZeroResultsRate"] = "Search Zero Results Rate";
49
- SpiffyMetricsEventName$1["SearchFilterClicked"] = "Search Filter Clicked";
50
- SpiffyMetricsEventName$1["SearchSortClicked"] = "Search Sort Clicked";
51
- return SpiffyMetricsEventName$1;
52
- }({});
53
- const AmplitudeContext = (0, react.createContext)(null);
54
- const AmplitudeProvider = ({ children }) => {
55
- const userId = (0, jotai.useAtomValue)(require_app.userIdAtom);
56
- const amplitudeApiKey = (0, jotai.useAtomValue)(require_enviveConfig.amplitudeApiKeyAtom);
57
- const dataResidency = (0, jotai.useAtomValue)(require_enviveConfig.dataResidencyAtom);
58
- const orgGaConfig = (0, jotai.useAtomValue)(require_orgAnalyticsConfig.orgAnalyticsGoogleAnalyticsConfigAtom);
59
- const env = (0, jotai.useAtomValue)(require_enviveConfig.envAtom);
60
- const contextSource = (0, jotai.useAtomValue)(require_enviveConfig.contextSourceAtom);
61
- (0, jotai.useAtomValue)(require_enviveConfig.identifyingPrefixAtom);
62
- const { getItem } = require_localStorageContext.useLocalStorage();
63
- const { publicKey, featureOverrides, variantUrlOverride, variantInfoOverride, show, orgShortName, featureGates } = require_enviveConfigContext.useEnviveConfig();
64
- const { featureFlagService } = require_featureFlagServiceContext.useFeatureFlagService();
65
- const [amplitudeClient, setAmplitudeClient] = react.default.useState(void 0);
66
- const [internalEventTrackingEnrichment, setInternalEventTrackingEnrichment] = react.default.useState(void 0);
67
- const [supplementalDefaultProps, setSupplementalDefaultProps] = react.default.useState({});
68
- const isReady = Boolean(userId && featureFlagService && amplitudeApiKey && userId);
69
- const getDefaultTrackingProps = (0, react.useCallback)(() => {
70
- const gatesProps = featureGates ? featureGates.reduce((acc, curr) => {
71
- if (curr.name && curr.value != null) return {
72
- ...acc,
73
- [`feature_gate.${curr.name}`]: curr.value
74
- };
75
- return acc;
76
- }, {}) : {};
77
- const experimentProps = {};
78
- return {
79
- ...gatesProps,
80
- ...experimentProps,
81
- ...supplementalDefaultProps,
82
- app_id: "commerce-chat-react-component",
83
- chat_id: getItem(require_localStorageContext.LocalStorageKeys.ChatId),
84
- env: env || "unknown",
85
- app_source: contextSource,
86
- "org.short_name": orgShortName,
87
- "user.id": userId,
88
- "cdp.user_id": null,
89
- "cdp.provider": null,
90
- "event.source": "web-browser",
91
- "event.type": "user-activity",
92
- "event.id": null,
93
- "event.channel": "web",
94
- "event.timestamp": null
95
- };
96
- }, [
97
- featureGates,
98
- supplementalDefaultProps,
99
- env,
100
- contextSource,
101
- orgShortName,
102
- userId
103
- ]);
104
- const eventPropsToPrefixedEventProps = (0, react.useCallback)((eventName, eventProps) => {
105
- const prefix = eventName.toLowerCase().replace(/\s+/g, "_");
106
- return Object.entries(eventProps).reduce((acc, [key, value$1]) => {
107
- acc[`${prefix}.${key}`] = value$1;
108
- return acc;
109
- }, {});
110
- }, []);
111
- const amplitudeSessionReplayInit = (0, react.useCallback)(() => {
112
- const isEnabled = Boolean(orgShortName === require_models.OrgShortName.UniqueVintage && featureFlagService?.isClientSessionEnabled() && featureFlagService?.isFeatureGateEnabled(require_featureGates.FeatureGates.IsNewFeatureEnabled));
113
- const sampleRate = 1;
114
- try {
115
- require_logger.logger_default.logDebug(`[spiffy-ai] amplitude session-replay initializing isEnabled=${isEnabled} sampleRate=${sampleRate}`);
116
- if (!isEnabled) return isEnabled;
117
- return isEnabled;
118
- } catch (e) {
119
- require_logger.logger_default.logError("[spiffy-ai] Error initializing amplitude session-replay", e);
120
- return false;
121
- }
122
- }, [orgShortName, featureFlagService]);
123
- const getEventTrackingEnrichment = (0, react.useCallback)(() => {
124
- if (internalEventTrackingEnrichment !== void 0) return internalEventTrackingEnrichment;
125
- const enrichment = {
126
- name: "page-view-tracking-enrichment",
127
- type: "enrichment",
128
- setup: async () => void 0,
129
- execute: async (event) => {
130
- let enrichedEvent;
131
- if (["[Amplitude] Page Viewed", `[Spiffy] ${SpiffyMetricsEventName.BundleLoaded}`].includes(event.event_type)) {
132
- const globalProperties = {};
133
- if (publicKey) globalProperties["globalProperties.publicKey"] = publicKey;
134
- if (featureOverrides) Object.entries(featureOverrides).forEach(([key, value$1]) => {
135
- globalProperties[`globalProperties.featureOverrides.${key}`] = String(value$1);
136
- });
137
- if (variantUrlOverride) globalProperties["globalProperties.variantUrlOverride"] = variantUrlOverride;
138
- if (variantInfoOverride) globalProperties["globalProperties.variantInfoOverride"] = JSON.stringify(variantInfoOverride);
139
- if (show != null) globalProperties["globalProperties.show"] = String(show);
140
- const enabledFeatures = featureFlagService.getFeatureFlags();
141
- const enabledFeaturesProperties = Object.entries(enabledFeatures).reduce((acc, [key, value$1]) => ({
142
- ...acc,
143
- [`enabledFeatures.${key}`]: `${value$1}`
144
- }), {});
145
- const timingProperties = { "timing.enriched_at_ms": window.performance?.now() };
146
- enrichedEvent = {
147
- ...event,
148
- event_properties: {
149
- ...event.event_properties,
150
- ...getDefaultTrackingProps(),
151
- ...globalProperties,
152
- ...enabledFeaturesProperties,
153
- ...timingProperties
154
- }
155
- };
156
- } else enrichedEvent = event;
157
- require_events.EventsDispatcher.dispatch(require_events.SpiffyEvent.AMPLITUDE_EVENT, enrichedEvent);
158
- return enrichedEvent;
159
- }
160
- };
161
- setInternalEventTrackingEnrichment(enrichment);
162
- return enrichment;
163
- }, [
164
- internalEventTrackingEnrichment,
165
- getDefaultTrackingProps,
166
- featureFlagService,
167
- publicKey,
168
- featureOverrides,
169
- variantUrlOverride,
170
- variantInfoOverride,
171
- show
172
- ]);
173
- (0, react.useEffect)(() => {
174
- if (isReady && !amplitudeClient) {
175
- const currentAmplitudeInstance = (0, __amplitude_analytics_browser.createInstance)();
176
- const isSessionsEnabled = amplitudeSessionReplayInit();
177
- currentAmplitudeInstance.add(getEventTrackingEnrichment());
178
- currentAmplitudeInstance.init(amplitudeApiKey, userId, {
179
- serverZone: dataResidency,
180
- trackingOptions: { ipAddress: true },
181
- autocapture: {
182
- attribution: true,
183
- pageViews: { trackHistoryChanges: "pathOnly" },
184
- sessions: isSessionsEnabled,
185
- formInteractions: false,
186
- fileDownloads: false
187
- }
188
- });
189
- setAmplitudeClient(currentAmplitudeInstance);
190
- }
191
- }, [
192
- isReady,
193
- amplitudeClient,
194
- amplitudeApiKey,
195
- userId,
196
- dataResidency,
197
- amplitudeSessionReplayInit,
198
- getEventTrackingEnrichment
199
- ]);
200
- const trackEvent = (0, react.useCallback)(async ({ eventName, eventProps, eventGroups, alsoSendToGoogleAnalytics = false }) => {
201
- require_logger.logger_default.logDebug("Submitting event", eventName);
202
- try {
203
- const decoratedEventName = `[Spiffy] ${eventName}`;
204
- if (!amplitudeClient) {
205
- require_logger.logger_default.logWarn("amplitude client undefined", void 0, { event_name: decoratedEventName });
206
- return;
207
- }
208
- const eventData = JSON.stringify({
209
- eventName,
210
- eventProps,
211
- created_at: (/* @__PURE__ */ new Date()).toISOString()
212
- });
213
- const data = new TextEncoder().encode(eventData);
214
- const hashBuffer = await crypto.subtle.digest("SHA-256", data);
215
- const currentInsertId = Array.from(new Uint8Array(hashBuffer)).map((b) => b.toString(16).padStart(2, "0")).join("");
216
- require_logger.logger_default.logDebug(`amplitude tracking ${decoratedEventName}`, null, {
217
- event_name: decoratedEventName,
218
- props: eventProps
219
- });
220
- amplitudeClient.track(decoratedEventName, {
221
- ...getDefaultTrackingProps(),
222
- ...eventProps,
223
- ...eventProps ? eventPropsToPrefixedEventProps(eventName, eventProps) : {}
224
- }, {
225
- ...eventGroups,
226
- insert_id: currentInsertId
227
- });
228
- if (alsoSendToGoogleAnalytics && orgGaConfig) {
229
- require_logger.logger_default.logDebug("[spiffy-ai] GA tracking", decoratedEventName);
230
- if (window.dataLayer) window.dataLayer.push({
231
- event: decoratedEventName,
232
- eventProps
233
- });
234
- }
235
- } catch (err) {
236
- require_logger.logger_default.logError("[spiffy-ai] Error tracking event", err, {
237
- eventName,
238
- eventProps
239
- });
240
- }
241
- }, [
242
- amplitudeClient,
243
- getDefaultTrackingProps,
244
- eventPropsToPrefixedEventProps,
245
- orgGaConfig
246
- ]);
247
- const value = (0, react.useMemo)(() => ({
248
- trackEvent,
249
- isReady,
250
- setSupplementalDefaultProps: (props) => setSupplementalDefaultProps(props)
251
- }), [
252
- trackEvent,
253
- isReady,
254
- setSupplementalDefaultProps
255
- ]);
256
- return /* @__PURE__ */ (0, react_jsx_runtime.jsx)(AmplitudeContext.Provider, {
257
- value,
258
- children
259
- });
260
- };
261
- const useAmplitude = () => {
262
- const context = (0, react.useContext)(AmplitudeContext);
263
- if (!context) throw new Error("useAmplitude must be used within AmplitudeProvider");
264
- return context;
265
- };
266
-
267
- //#endregion
268
- Object.defineProperty(exports, 'AmplitudeProvider', {
269
- enumerable: true,
270
- get: function () {
271
- return AmplitudeProvider;
272
- }
273
- });
274
- Object.defineProperty(exports, 'SpiffyMetricsEventName', {
275
- enumerable: true,
276
- get: function () {
277
- return SpiffyMetricsEventName;
278
- }
279
- });
280
- Object.defineProperty(exports, 'useAmplitude', {
281
- enumerable: true,
282
- get: function () {
283
- return useAmplitude;
284
- }
285
- });
286
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"amplitudeContext-C8tT74Mi.cjs","names":["AmplitudeProvider: React.FC<{ children: React.ReactNode }>","userIdAtom","amplitudeApiKeyAtom","dataResidencyAtom","orgAnalyticsGoogleAnalyticsConfigAtom","envAtom","contextSourceAtom","identifyingPrefixAtom","useLocalStorage","useEnviveConfig","useFeatureFlagService","React","LocalStorageKeys","value","OrgShortName","FeatureGates","enrichment: EnrichmentPlugin","enrichedEvent: Event","globalProperties: Record<string, string>","SpiffyEvent","currentAmplitudeInstance: BrowserClient"],"sources":["../src/contexts/amplitudeContext/amplitudeContext.tsx"],"sourcesContent":["import React, {\n  createContext,\n  useContext,\n  useCallback,\n  useMemo,\n  useEffect,\n} from \"react\";\nimport { useAtomValue } from \"jotai\";\nimport { createInstance } from \"@amplitude/analytics-browser\";\nimport { FeatureGates } from \"src/application/models/featureGates\";\nimport { OrgShortName } from \"src/application/models\";\nimport { EventsDispatcher, SpiffyEvent } from \"src/events\";\nimport {\n  amplitudeApiKeyAtom,\n  contextSourceAtom,\n  dataResidencyAtom,\n  identifyingPrefixAtom,\n} from \"src/atoms/envive/enviveConfig\";\nimport {\n  LocalStorageKeys,\n  useLocalStorage,\n} from \"src/contexts/localStorageContext\";\nimport { orgAnalyticsGoogleAnalyticsConfigAtom } from \"src/atoms/org/orgAnalyticsConfig\";\nimport { userIdAtom } from \"src/atoms/app\";\nimport { envAtom } from \"src/atoms/envive/enviveConfig\";\nimport Logger from \"src/application/logging/logger\";\nimport type {\n  BrowserClient,\n  EnrichmentPlugin,\n  Event,\n  ServerZoneType,\n} from \"@amplitude/analytics-types\";\nimport { useEnviveConfig } from \"src/contexts/enviveConfigContext/enviveConfigContext\";\nimport {\n  useFeatureFlagService,\n} from \"src/contexts/featureFlagServiceContext/featureFlagServiceContext\";\nimport { useUserIdentity } from \"src/contexts/userIdentityContext/userIdentityContext\";\n\nexport enum SpiffyMetricsEventName {\n  BundleLoaded = \"Bundle Loaded\",\n  ChatLiveAgentBtnClick = \"Chat Live Agent Btn Click\",\n  ChatFloatingButtonVisible = \"Chat Floating Button Visible\",\n  ChatComponentVisible = \"Chat Component Visible\",\n  ChatComponentExpanded = \"Chat Component Expanded\",\n  ChatComponentCollapsed = \"Chat Component Collapsed\",\n  ChatUserMessageInput = \"Chat User Message Input\",\n  ChatSuggestionClicked = \"Chat Suggestion Clicked\",\n  ChatAssistantResponse = \"Chat Assistant Response\",\n  ProductCardClicked = \"Product Card Clicked\",\n  ProductReviewCardClicked = \"Product Review Card Clicked\",\n  AddToCartClicked = \"Add to Cart Clicked\",\n  PromptCardClicked = \"Prompt Card Clicked\",\n  SupportedEvent = \"Supported Event\",\n  SearchBackToResponseClicked = \"Search Back to Response Clicked\",\n  PerformanceMetrics = \"Performance Metrics\",\n  SearchBarClicked = \"Search Bar Clicked\",\n  OrderLookupStarted = \"Order Lookup Started\",\n  OrderLookupFormSubmitted = \"Order Lookup Form Submitted\",\n  SearchComponentVisible = \"Search Component Visible\",\n  SearchZeroStateSuggestionClicked = \"Search Zero State Suggestion Clicked\", // This is the scrolling list of suggestion buttons in global search\n  SearchInputStarted = \"Search Input Started\",\n  SearchQuerySubmitted = \"Search Query Submitted\",\n  // SearchAutocompleteViewed = 'Search Autocomplete Viewed', // TODO: add this when autocomplete is added\n  // SearchAutocompleteClicked = 'Search Autocomplete Clicked', // TODO: add this when autocomplete is added\n  SearchResultsViewed = \"Search Results Viewed\",\n  SearchTimeToFirstClick = \"Search Time to First Click\",\n  SearchZeroResultsRate = \"Search Zero Results Rate\",\n  SearchFilterClicked = \"Search Filter Clicked\",\n  SearchSortClicked = \"Search Sort Clicked\",\n}\n\ninterface TrackEventParams {\n  eventName: SpiffyMetricsEventName;\n  eventProps?: Record<string, unknown>;\n  eventGroups?: Record<string, unknown>;\n  alsoSendToGoogleAnalytics?: boolean;\n}\n\ninterface AmplitudeContextType {\n  trackEvent: (params: TrackEventParams) => Promise<void>;\n  isReady: boolean;\n  setSupplementalDefaultProps: (props: Record<string, unknown>) => void;\n}\n\nconst AmplitudeContext = createContext<AmplitudeContextType | null>(null);\n\nexport const AmplitudeProvider: React.FC<{ children: React.ReactNode }> = ({\n  children,\n}) => {\n  const userId = useAtomValue(userIdAtom);\n  const amplitudeApiKey = useAtomValue(amplitudeApiKeyAtom);\n  const dataResidency = useAtomValue(dataResidencyAtom);\n  const orgGaConfig = useAtomValue(orgAnalyticsGoogleAnalyticsConfigAtom);\n  const env = useAtomValue(envAtom);\n  const contextSource = useAtomValue(contextSourceAtom);\n  const identifyingPrefix = useAtomValue(identifyingPrefixAtom);\n  const { getItem } = useLocalStorage();\n  const {\n    publicKey,\n    featureOverrides,\n    variantUrlOverride,\n    variantInfoOverride,\n    show,\n    orgShortName,\n    featureGates,\n  } = useEnviveConfig();\n\n  const { featureFlagService } = useFeatureFlagService();\n\n  const [amplitudeClient, setAmplitudeClient] = React.useState<\n    BrowserClient | undefined\n  >(undefined);\n  const [internalEventTrackingEnrichment, setInternalEventTrackingEnrichment] =\n    React.useState<EnrichmentPlugin | undefined>(undefined);\n  const [supplementalDefaultProps, setSupplementalDefaultProps] =\n    React.useState<Record<string, unknown>>({});\n\n  const isReady = Boolean(\n    userId && featureFlagService && amplitudeApiKey && userId\n  );\n\n  const getDefaultTrackingProps = useCallback((): Record<string, unknown> => {\n    const gatesProps = featureGates\n      ? featureGates.reduce<Record<string, boolean>>((acc, curr) => {\n          if (curr.name && curr.value != null) {\n            return { ...acc, [`feature_gate.${curr.name}`]: curr.value };\n          }\n          return acc;\n        }, {})\n      : {};\n    const experimentProps = {}; // No direct equivalent for experiments in EnviveConfig yet\n\n    const orgLevelAmplitudeTrackingProps = {\n      ...gatesProps,\n      ...experimentProps,\n    };\n\n    return {\n      ...orgLevelAmplitudeTrackingProps,\n      ...supplementalDefaultProps,\n      // TODO: org_id is not directly available in EnviveConfig. Need to find a new source or derive it.\n      // org_id: orgConfig?.org?.org?.id,\n      app_id: \"commerce-chat-react-component\",\n      chat_id: getItem(LocalStorageKeys.ChatId),\n      env: env || \"unknown\",\n      app_source: contextSource,\n      \"org.short_name\": orgShortName,\n      \"user.id\": userId,\n      \"cdp.user_id\": null,\n      \"cdp.provider\": null,\n      \"event.source\": \"web-browser\",\n      \"event.type\": \"user-activity\",\n      \"event.id\": null,\n      \"event.channel\": \"web\",\n      \"event.timestamp\": null,\n    };\n  }, [\n    featureGates,\n    supplementalDefaultProps,\n    env,\n    contextSource,\n    orgShortName,\n    userId,\n  ]);\n\n  const eventPropsToPrefixedEventProps = useCallback(\n    (\n      eventName: SpiffyMetricsEventName,\n      eventProps: Record<string, unknown>\n    ): Record<string, unknown> => {\n      const prefix = eventName.toLowerCase().replace(/\\s+/g, \"_\");\n      return Object.entries(eventProps).reduce((acc, [key, value]) => {\n        acc[`${prefix}.${key}`] = value;\n        return acc;\n      }, {} as Record<string, unknown>);\n    },\n    []\n  );\n\n  const amplitudeSessionReplayInit = useCallback((): boolean => {\n    const isEnabled = Boolean(\n      orgShortName === OrgShortName.UniqueVintage &&\n        featureFlagService?.isClientSessionEnabled() &&\n        featureFlagService?.isFeatureGateEnabled(\n          FeatureGates.IsNewFeatureEnabled\n        )\n    );\n    const sampleRate = 1;\n\n    try {\n      Logger.logDebug(\n        `[spiffy-ai] amplitude session-replay initializing isEnabled=${isEnabled} sampleRate=${sampleRate}`\n      );\n\n      if (!isEnabled) {\n        return isEnabled;\n      }\n\n      // amplitudeInstance.add(sessionReplayPlugin({ sampleRate }));\n\n      return isEnabled;\n    } catch (e) {\n      Logger.logError(\n        \"[spiffy-ai] Error initializing amplitude session-replay\",\n        e\n      );\n      return false;\n    }\n  }, [orgShortName, featureFlagService]);\n\n  const getEventTrackingEnrichment = useCallback((): EnrichmentPlugin => {\n    if (internalEventTrackingEnrichment !== undefined) {\n      return internalEventTrackingEnrichment;\n    }\n\n    const enrichment: EnrichmentPlugin = {\n      name: \"page-view-tracking-enrichment\",\n      type: \"enrichment\",\n      setup: async () => undefined,\n      execute: async (event: Event): Promise<Event> => {\n        let enrichedEvent: Event;\n\n        const eventsToEnrich = [\n          \"[Amplitude] Page Viewed\",\n          `[Spiffy] ${SpiffyMetricsEventName.BundleLoaded}`,\n        ];\n\n        if (eventsToEnrich.includes(event.event_type)) {\n          const globalProperties: Record<string, string> = {};\n\n          if (publicKey) {\n            globalProperties[\"globalProperties.publicKey\"] = publicKey;\n          }\n          if (featureOverrides) {\n            Object.entries(featureOverrides).forEach(([key, value]) => {\n              globalProperties[`globalProperties.featureOverrides.${key}`] =\n                String(value);\n            });\n          }\n          if (variantUrlOverride) {\n            globalProperties[\"globalProperties.variantUrlOverride\"] =\n              variantUrlOverride;\n          }\n          if (variantInfoOverride) {\n            globalProperties[\"globalProperties.variantInfoOverride\"] =\n              JSON.stringify(variantInfoOverride);\n          }\n          if (show != null) {\n            globalProperties[\"globalProperties.show\"] = String(show);\n          }\n\n          const enabledFeatures = featureFlagService!.getFeatureFlags();\n          const enabledFeaturesProperties = Object.entries(\n            enabledFeatures\n          ).reduce<Record<string, string>>(\n            (acc, [key, value]) => ({\n              ...acc,\n              [`enabledFeatures.${key}`]: `${value}`,\n            }),\n            {}\n          );\n\n          const timingProperties = {\n            \"timing.enriched_at_ms\": window.performance?.now(),\n          };\n\n          enrichedEvent = {\n            ...event,\n            event_properties: {\n              ...event.event_properties,\n              ...getDefaultTrackingProps(),\n              ...globalProperties,\n              ...enabledFeaturesProperties,\n              ...timingProperties,\n            },\n          };\n        } else {\n          enrichedEvent = event;\n        }\n\n        EventsDispatcher.dispatch(SpiffyEvent.AMPLITUDE_EVENT, enrichedEvent);\n\n        return enrichedEvent;\n      },\n    };\n    setInternalEventTrackingEnrichment(enrichment);\n    return enrichment;\n  }, [\n    internalEventTrackingEnrichment,\n    getDefaultTrackingProps,\n    featureFlagService,\n    publicKey,\n    featureOverrides,\n    variantUrlOverride,\n    variantInfoOverride,\n    show,\n  ]);\n\n  useEffect(() => {\n    if (isReady && !amplitudeClient) {\n      const currentAmplitudeInstance: BrowserClient = createInstance();\n      const isSessionsEnabled = amplitudeSessionReplayInit();\n      currentAmplitudeInstance.add(getEventTrackingEnrichment());\n      currentAmplitudeInstance.init(amplitudeApiKey!, userId!, {\n        serverZone: dataResidency as ServerZoneType,\n        trackingOptions: {\n          ipAddress: true,\n        },\n        autocapture: {\n          attribution: true,\n          pageViews: {\n            trackHistoryChanges: \"pathOnly\",\n          },\n          sessions: isSessionsEnabled,\n          formInteractions: false,\n          fileDownloads: false,\n        },\n      });\n      setAmplitudeClient(currentAmplitudeInstance);\n    }\n  }, [\n    isReady,\n    amplitudeClient,\n    amplitudeApiKey,\n    userId,\n    dataResidency,\n    amplitudeSessionReplayInit,\n    getEventTrackingEnrichment,\n  ]);\n\n  const trackEvent = useCallback(\n    async ({\n      eventName,\n      eventProps,\n      eventGroups,\n      alsoSendToGoogleAnalytics = false,\n    }: TrackEventParams): Promise<void> => {\n      Logger.logDebug(\"Submitting event\", eventName);\n      try {\n        const decoratedEventName = `[Spiffy] ${eventName}`;\n\n        if (!amplitudeClient) {\n          Logger.logWarn(\"amplitude client undefined\", undefined, {\n            event_name: decoratedEventName,\n          });\n          return;\n        }\n\n        const eventData = JSON.stringify({\n          eventName,\n          eventProps,\n          created_at: new Date().toISOString(),\n        });\n        const encoder = new TextEncoder();\n        const data = encoder.encode(eventData);\n         // calculate a hash of the event properties to use as the insert_id so that duplicate events\n        // are automatically dropped by Amplitude\n        const hashBuffer = await crypto.subtle.digest(\"SHA-256\", data);\n        const hashArray = Array.from(new Uint8Array(hashBuffer));\n        const currentInsertId = hashArray\n          .map((b) => b.toString(16).padStart(2, \"0\"))\n          .join(\"\");\n\n        Logger.logDebug(`amplitude tracking ${decoratedEventName}`, null, {\n          event_name: decoratedEventName,\n          props: eventProps,\n        });\n\n        amplitudeClient.track(\n          decoratedEventName,\n          {\n            ...getDefaultTrackingProps(),\n            ...eventProps,\n            ...(eventProps\n              ? eventPropsToPrefixedEventProps(eventName, eventProps)\n              : {}),\n          },\n          {\n            ...eventGroups,\n            insert_id: currentInsertId,\n          }\n        );\n\n        if (alsoSendToGoogleAnalytics && orgGaConfig) {\n          // TODO: Add in windowDataLayerService or context alternative and hook it up here\n          Logger.logDebug(\"[spiffy-ai] GA tracking\", decoratedEventName);\n          if (window.dataLayer) {\n            (window.dataLayer as any[]).push({\n              event: decoratedEventName,\n              eventProps: eventProps,\n            });\n          }\n        }\n      } catch (err) {\n        Logger.logError(\"[spiffy-ai] Error tracking event\", err, {\n          eventName,\n          eventProps,\n        });\n      }\n    },\n    [\n      amplitudeClient,\n      getDefaultTrackingProps,\n      eventPropsToPrefixedEventProps,\n      orgGaConfig,\n    ]\n  );\n\n  const value = useMemo(\n    () => ({\n      trackEvent,\n      isReady,\n      setSupplementalDefaultProps: (props: Record<string, unknown>) =>\n        setSupplementalDefaultProps(props),\n    }),\n    [trackEvent, isReady, setSupplementalDefaultProps]\n  );\n\n  return (\n    <AmplitudeContext.Provider value={value}>\n      {children}\n    </AmplitudeContext.Provider>\n  );\n};\n\nexport const useAmplitude = () => {\n  const context = useContext(AmplitudeContext);\n  if (!context) {\n    throw new Error(\"useAmplitude must be used within AmplitudeProvider\");\n  }\n  return context;\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;AAsCA,IAAY,4EAAL;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;;;AAgBF,MAAM,4CAA8D,KAAK;AAEzE,MAAaA,qBAA8D,EACzE,eACI;CACJ,MAAM,iCAAsBC,uBAAW;CACvC,MAAM,0CAA+BC,yCAAoB;CACzD,MAAM,wCAA6BC,uCAAkB;CACrD,MAAM,sCAA2BC,iEAAsC;CACvE,MAAM,8BAAmBC,6BAAQ;CACjC,MAAM,wCAA6BC,uCAAkB;AAC3B,yBAAaC,2CAAsB;CAC7D,MAAM,EAAE,YAAYC,6CAAiB;CACrC,MAAM,EACJ,WACA,kBACA,oBACA,qBACA,MACA,cACA,iBACEC,6CAAiB;CAErB,MAAM,EAAE,uBAAuBC,yDAAuB;CAEtD,MAAM,CAAC,iBAAiB,sBAAsBC,cAAM,SAElD,OAAU;CACZ,MAAM,CAAC,iCAAiC,sCACtCA,cAAM,SAAuC,OAAU;CACzD,MAAM,CAAC,0BAA0B,+BAC/BA,cAAM,SAAkC,EAAE,CAAC;CAE7C,MAAM,UAAU,QACd,UAAU,sBAAsB,mBAAmB,OACpD;CAED,MAAM,uDAAqE;EACzE,MAAM,aAAa,eACf,aAAa,QAAiC,KAAK,SAAS;AAC1D,OAAI,KAAK,QAAQ,KAAK,SAAS,KAC7B,QAAO;IAAE,GAAG;KAAM,gBAAgB,KAAK,SAAS,KAAK;IAAO;AAE9D,UAAO;KACN,EAAE,CAAC,GACN,EAAE;EACN,MAAM,kBAAkB,EAAE;AAO1B,SAAO;GAJL,GAAG;GACH,GAAG;GAKH,GAAG;GAGH,QAAQ;GACR,SAAS,QAAQC,6CAAiB,OAAO;GACzC,KAAK,OAAO;GACZ,YAAY;GACZ,kBAAkB;GAClB,WAAW;GACX,eAAe;GACf,gBAAgB;GAChB,gBAAgB;GAChB,cAAc;GACd,YAAY;GACZ,iBAAiB;GACjB,mBAAmB;GACpB;IACA;EACD;EACA;EACA;EACA;EACA;EACA;EACD,CAAC;CAEF,MAAM,yDAEF,WACA,eAC4B;EAC5B,MAAM,SAAS,UAAU,aAAa,CAAC,QAAQ,QAAQ,IAAI;AAC3D,SAAO,OAAO,QAAQ,WAAW,CAAC,QAAQ,KAAK,CAAC,KAAKC,aAAW;AAC9D,OAAI,GAAG,OAAO,GAAG,SAASA;AAC1B,UAAO;KACN,EAAE,CAA4B;IAEnC,EAAE,CACH;CAED,MAAM,0DAAwD;EAC5D,MAAM,YAAY,QAChB,iBAAiBC,4BAAa,iBAC5B,oBAAoB,wBAAwB,IAC5C,oBAAoB,qBAClBC,kCAAa,oBACd,CACJ;EACD,MAAM,aAAa;AAEnB,MAAI;AACF,iCAAO,SACL,+DAA+D,UAAU,cAAc,aACxF;AAED,OAAI,CAAC,UACH,QAAO;AAKT,UAAO;WACA,GAAG;AACV,iCAAO,SACL,2DACA,EACD;AACD,UAAO;;IAER,CAAC,cAAc,mBAAmB,CAAC;CAEtC,MAAM,0DAAiE;AACrE,MAAI,oCAAoC,OACtC,QAAO;EAGT,MAAMC,aAA+B;GACnC,MAAM;GACN,MAAM;GACN,OAAO,YAAY;GACnB,SAAS,OAAO,UAAiC;IAC/C,IAAIC;AAOJ,QALuB,CACrB,2BACA,YAAY,uBAAuB,eACpC,CAEkB,SAAS,MAAM,WAAW,EAAE;KAC7C,MAAMC,mBAA2C,EAAE;AAEnD,SAAI,UACF,kBAAiB,gCAAgC;AAEnD,SAAI,iBACF,QAAO,QAAQ,iBAAiB,CAAC,SAAS,CAAC,KAAKL,aAAW;AACzD,uBAAiB,qCAAqC,SACpD,OAAOA,QAAM;OACf;AAEJ,SAAI,mBACF,kBAAiB,yCACf;AAEJ,SAAI,oBACF,kBAAiB,0CACf,KAAK,UAAU,oBAAoB;AAEvC,SAAI,QAAQ,KACV,kBAAiB,2BAA2B,OAAO,KAAK;KAG1D,MAAM,kBAAkB,mBAAoB,iBAAiB;KAC7D,MAAM,4BAA4B,OAAO,QACvC,gBACD,CAAC,QACC,KAAK,CAAC,KAAKA,cAAY;MACtB,GAAG;OACF,mBAAmB,QAAQ,GAAGA;MAChC,GACD,EAAE,CACH;KAED,MAAM,mBAAmB,EACvB,yBAAyB,OAAO,aAAa,KAAK,EACnD;AAED,qBAAgB;MACd,GAAG;MACH,kBAAkB;OAChB,GAAG,MAAM;OACT,GAAG,yBAAyB;OAC5B,GAAG;OACH,GAAG;OACH,GAAG;OACJ;MACF;UAED,iBAAgB;AAGlB,oCAAiB,SAASM,2BAAY,iBAAiB,cAAc;AAErE,WAAO;;GAEV;AACD,qCAAmC,WAAW;AAC9C,SAAO;IACN;EACD;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD,CAAC;AAEF,4BAAgB;AACd,MAAI,WAAW,CAAC,iBAAiB;GAC/B,MAAMC,8EAA0D;GAChE,MAAM,oBAAoB,4BAA4B;AACtD,4BAAyB,IAAI,4BAA4B,CAAC;AAC1D,4BAAyB,KAAK,iBAAkB,QAAS;IACvD,YAAY;IACZ,iBAAiB,EACf,WAAW,MACZ;IACD,aAAa;KACX,aAAa;KACb,WAAW,EACT,qBAAqB,YACtB;KACD,UAAU;KACV,kBAAkB;KAClB,eAAe;KAChB;IACF,CAAC;AACF,sBAAmB,yBAAyB;;IAE7C;EACD;EACA;EACA;EACA;EACA;EACA;EACA;EACD,CAAC;CAEF,MAAM,oCACJ,OAAO,EACL,WACA,YACA,aACA,4BAA4B,YACS;AACrC,gCAAO,SAAS,oBAAoB,UAAU;AAC9C,MAAI;GACF,MAAM,qBAAqB,YAAY;AAEvC,OAAI,CAAC,iBAAiB;AACpB,kCAAO,QAAQ,8BAA8B,QAAW,EACtD,YAAY,oBACb,CAAC;AACF;;GAGF,MAAM,YAAY,KAAK,UAAU;IAC/B;IACA;IACA,6BAAY,IAAI,MAAM,EAAC,aAAa;IACrC,CAAC;GAEF,MAAM,OADU,IAAI,aAAa,CACZ,OAAO,UAAU;GAGtC,MAAM,aAAa,MAAM,OAAO,OAAO,OAAO,WAAW,KAAK;GAE9D,MAAM,kBADY,MAAM,KAAK,IAAI,WAAW,WAAW,CAAC,CAErD,KAAK,MAAM,EAAE,SAAS,GAAG,CAAC,SAAS,GAAG,IAAI,CAAC,CAC3C,KAAK,GAAG;AAEX,iCAAO,SAAS,sBAAsB,sBAAsB,MAAM;IAChE,YAAY;IACZ,OAAO;IACR,CAAC;AAEF,mBAAgB,MACd,oBACA;IACE,GAAG,yBAAyB;IAC5B,GAAG;IACH,GAAI,aACA,+BAA+B,WAAW,WAAW,GACrD,EAAE;IACP,EACD;IACE,GAAG;IACH,WAAW;IACZ,CACF;AAED,OAAI,6BAA6B,aAAa;AAE5C,kCAAO,SAAS,2BAA2B,mBAAmB;AAC9D,QAAI,OAAO,UACT,CAAC,OAAO,UAAoB,KAAK;KAC/B,OAAO;KACK;KACb,CAAC;;WAGC,KAAK;AACZ,iCAAO,SAAS,oCAAoC,KAAK;IACvD;IACA;IACD,CAAC;;IAGN;EACE;EACA;EACA;EACA;EACD,CACF;CAED,MAAM,kCACG;EACL;EACA;EACA,8BAA8B,UAC5B,4BAA4B,MAAM;EACrC,GACD;EAAC;EAAY;EAAS;EAA4B,CACnD;AAED,QACE,2CAAC,iBAAiB;EAAgB;EAC/B;GACyB;;AAIhC,MAAa,qBAAqB;CAChC,MAAM,gCAAqB,iBAAiB;AAC5C,KAAI,CAAC,QACH,OAAM,IAAI,MAAM,qDAAqD;AAEvE,QAAO"}