@envive-ai/react-hooks 0.3.21 → 0.3.23

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 (111) hide show
  1. package/dist/application/models/featureGates.cjs +2 -1
  2. package/dist/application/models/featureGates.d.cts +2 -1
  3. package/dist/application/models/featureGates.d.ts +2 -1
  4. package/dist/application/models/featureGates.js +2 -1
  5. package/dist/atoms/app/index.d.cts +7 -7
  6. package/dist/atoms/app/variant.d.cts +6 -6
  7. package/dist/atoms/app/variant.d.ts +6 -6
  8. package/dist/atoms/chat/chatState.cjs +3 -1
  9. package/dist/atoms/chat/chatState.d.cts +22 -19
  10. package/dist/atoms/chat/chatState.d.ts +22 -19
  11. package/dist/atoms/chat/chatState.js +3 -2
  12. package/dist/atoms/chat/form.d.cts +3 -3
  13. package/dist/atoms/chat/form.d.ts +2 -2
  14. package/dist/atoms/chat/index.cjs +1 -0
  15. package/dist/atoms/chat/index.d.cts +4 -4
  16. package/dist/atoms/chat/index.d.ts +4 -4
  17. package/dist/atoms/chat/index.js +2 -2
  18. package/dist/atoms/chat/lastMessage.d.cts +2 -2
  19. package/dist/atoms/chat/lastMessage.d.ts +2 -2
  20. package/dist/atoms/chat/messageQueue.d.cts +7 -7
  21. package/dist/atoms/chat/messageQueue.d.ts +6 -6
  22. package/dist/atoms/chat/performanceMetrics.d.cts +6 -6
  23. package/dist/atoms/chat/performanceMetrics.d.ts +6 -6
  24. package/dist/atoms/chat/renderedWidgetRefs.d.cts +3 -3
  25. package/dist/atoms/chat/renderedWidgetRefs.d.ts +2 -2
  26. package/dist/atoms/chat/replies.d.cts +3 -3
  27. package/dist/atoms/chat/replies.d.ts +2 -2
  28. package/dist/atoms/chat/suggestions.d.cts +3 -3
  29. package/dist/atoms/chat/suggestions.d.ts +2 -2
  30. package/dist/atoms/envive/enviveConfig.d.cts +13 -13
  31. package/dist/atoms/envive/enviveConfig.d.ts +1 -1
  32. package/dist/atoms/globalSearch/globalSearch.d.cts +5 -5
  33. package/dist/atoms/globalSearch/globalSearch.d.ts +5 -5
  34. package/dist/atoms/org/customerService.d.cts +6 -6
  35. package/dist/atoms/org/customerService.d.ts +6 -6
  36. package/dist/atoms/org/graphqlConfig.d.cts +4 -4
  37. package/dist/atoms/org/graphqlConfig.d.ts +4 -4
  38. package/dist/atoms/org/newOrgConfigAtom.d.cts +2 -2
  39. package/dist/atoms/org/newOrgConfigAtom.d.ts +2 -2
  40. package/dist/atoms/org/orgAnalyticsConfig.d.cts +5 -5
  41. package/dist/atoms/org/orgAnalyticsConfig.d.ts +5 -5
  42. package/dist/atoms/search/chatSearch.d.cts +17 -17
  43. package/dist/atoms/search/chatSearch.d.ts +17 -17
  44. package/dist/atoms/search/searchAPI.d.cts +13 -13
  45. package/dist/atoms/search/types.d.cts +1 -1
  46. package/dist/atoms/search/types.d.ts +1 -1
  47. package/dist/atoms/search/utils.d.ts +1 -1
  48. package/dist/atoms/widget/chatPreviewLoading.d.cts +2 -2
  49. package/dist/atoms/widget/chatPreviewLoading.d.ts +2 -2
  50. package/dist/contexts/amplitudeContext/amplitudeContext.cjs +9 -3
  51. package/dist/contexts/amplitudeContext/amplitudeContext.d.cts +2 -1
  52. package/dist/contexts/amplitudeContext/amplitudeContext.d.ts +2 -1
  53. package/dist/contexts/amplitudeContext/amplitudeContext.js +9 -3
  54. package/dist/contexts/enviveContext/enviveContext.cjs +3 -3
  55. package/dist/contexts/enviveContext/enviveContext.js +3 -3
  56. package/dist/contexts/enviveContext/types.d.ts +1 -1
  57. package/dist/contexts/hardcopyContext/hardcopyContext.cjs +5 -3
  58. package/dist/contexts/hardcopyContext/hardcopyContext.js +5 -3
  59. package/dist/contexts/salesAgentContext/chatAPI.cjs +12 -5
  60. package/dist/contexts/salesAgentContext/chatAPI.js +13 -6
  61. package/dist/contexts/systemSettingsContext/systemSettingsContext.d.cts +2 -2
  62. package/dist/contexts/types.d.cts +1 -1
  63. package/dist/contexts/typesV3.cjs +1 -1
  64. package/dist/contexts/typesV3.d.cts +2 -1
  65. package/dist/contexts/typesV3.d.ts +2 -1
  66. package/dist/contexts/typesV3.js +1 -1
  67. package/dist/hooks/GrabAndScroll/useGrabAndScroll.d.cts +2 -2
  68. package/dist/hooks/Search/useSearch.cjs +12 -4
  69. package/dist/hooks/Search/useSearch.js +12 -4
  70. package/dist/hooks/Search/useSearchInput.cjs +1 -1
  71. package/dist/hooks/Search/useSearchInput.js +1 -1
  72. package/dist/hooks/SystemSettingsContext/useSystemSettingsContext.d.cts +2 -2
  73. package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.cjs +26 -27
  74. package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.d.cts +8 -8
  75. package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.d.ts +8 -8
  76. package/dist/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.js +27 -28
  77. package/dist/hooks/WidgetInteraction/types.cjs +29 -1
  78. package/dist/hooks/WidgetInteraction/types.d.cts +17 -3
  79. package/dist/hooks/WidgetInteraction/types.d.ts +17 -3
  80. package/dist/hooks/WidgetInteraction/types.js +28 -2
  81. package/dist/hooks/WidgetInteraction/useWidgetInteraction.cjs +6 -2
  82. package/dist/hooks/WidgetInteraction/useWidgetInteraction.js +6 -2
  83. package/dist/hooks/utils.d.cts +1 -1
  84. package/dist/hooks/utils.d.ts +1 -1
  85. package/dist/services/amplitudeService/amplitudeService.cjs +9 -1
  86. package/dist/services/amplitudeService/amplitudeService.d.cts +2 -1
  87. package/dist/services/amplitudeService/amplitudeService.d.ts +2 -1
  88. package/dist/services/amplitudeService/amplitudeService.js +9 -1
  89. package/dist/services/ga4ProjectionService/ga4EventSchema.cjs +31 -27
  90. package/dist/services/ga4ProjectionService/ga4EventSchema.js +31 -27
  91. package/dist/services/ga4ProjectionService/ga4ProjectionService.cjs +31 -5
  92. package/dist/services/ga4ProjectionService/ga4ProjectionService.js +31 -5
  93. package/package.json +1 -1
  94. package/src/application/models/featureGates.ts +1 -0
  95. package/src/atoms/chat/chatState.ts +1 -0
  96. package/src/contexts/amplitudeContext/__tests__/amplitudeContext.test.tsx +31 -27
  97. package/src/contexts/amplitudeContext/amplitudeContext.tsx +5 -2
  98. package/src/contexts/hardcopyContext/hardcopyContext.tsx +10 -2
  99. package/src/contexts/pageContext/__tests__/pageContext.test.tsx +10 -0
  100. package/src/contexts/salesAgentContext/chatAPI.ts +6 -2
  101. package/src/contexts/typesV3.ts +1 -0
  102. package/src/hooks/Search/__tests__/useSearch.test.tsx +0 -4
  103. package/src/hooks/Search/useSearch.tsx +14 -8
  104. package/src/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent.ts +36 -35
  105. package/src/hooks/WidgetInteraction/types.ts +35 -2
  106. package/src/hooks/WidgetInteraction/useWidgetInteraction.ts +3 -1
  107. package/src/services/amplitudeService/__tests__/amplitudeService.test.ts +69 -6
  108. package/src/services/amplitudeService/amplitudeService.ts +13 -0
  109. package/src/services/ga4ProjectionService/__tests__/ga4ProjectionService.test.ts +110 -49
  110. package/src/services/ga4ProjectionService/ga4EventSchema.ts +35 -27
  111. package/src/services/ga4ProjectionService/ga4ProjectionService.ts +60 -6
@@ -24,31 +24,34 @@ import { EnviveMetricsEventName } from "../amplitudeService/eventNames.js";
24
24
  * IF IN DOUBT, DON'T HESITATE TO REACH OUT TO THE ANALYTICS TEAM!!!
25
25
  */
26
26
  const WIDGET_INTERACTION_DATA_PROJECTIONS = {
27
- widget_collapsed: { trigger_collapse_source: "collapse_source" },
28
- product_card_clicked: { trigger_product_id: "product_id" },
29
- suggestion_scrolled: { trigger_suggestion_id: "suggestion_id" }
27
+ widget_collapsed: { interaction_collapse_source: "widget_collapsed.collapse_source" },
28
+ product_card_clicked: { interaction_product_id: "product_card_clicked.product_id" },
29
+ suggestion_scrolled: { interaction_suggestion_id: "suggestion_scrolled.suggestion_id" }
30
30
  };
31
31
  const GA4_EVENT_SCHEMA = {
32
32
  [EnviveMetricsEventName.WidgetRendered]: {
33
33
  gaEventName: "envive_widget_rendered",
34
- allowedFields: [
35
- "context.page_type",
36
- "context.page_id",
37
- "trigger.widget",
38
- "trigger.widget_role",
39
- "context.surface"
40
- ]
34
+ allowedFields: [],
35
+ fieldProjections: {
36
+ page_type: "context.page_type",
37
+ page_id: "context.page_id",
38
+ surface: "context.surface",
39
+ widget: "trigger.widget",
40
+ widget_role: "trigger.widget_role"
41
+ }
41
42
  },
42
43
  [EnviveMetricsEventName.WidgetInteraction]: {
43
44
  gaEventName: "envive_widget_interaction",
44
- allowedFields: [
45
- "context.page_type",
46
- "context.page_id",
47
- "trigger.widget",
48
- "trigger.widget_interaction",
49
- "trigger.entity_role",
50
- "context.surface"
51
- ],
45
+ allowedFields: [],
46
+ fieldProjections: {
47
+ page_type: "context.page_type",
48
+ page_id: "context.page_id",
49
+ surface: "context.surface",
50
+ widget: "trigger.widget",
51
+ interaction_type: "trigger.widget_interaction",
52
+ interaction_class: "trigger.interaction_class",
53
+ widget_role: "trigger.widget_role"
54
+ },
52
55
  widgetInteractionDataProjections: WIDGET_INTERACTION_DATA_PROJECTIONS
53
56
  },
54
57
  [EnviveMetricsEventName.ChatRequest]: {
@@ -90,14 +93,15 @@ const GA4_EVENT_SCHEMA = {
90
93
  },
91
94
  [EnviveMetricsEventName.PageViewed]: {
92
95
  gaEventName: "envive_page_context_evaluated",
93
- allowedFields: [
94
- "context.page_type",
95
- "context.page_id",
96
- "context.supported",
97
- "context.ready",
98
- "context.page_variant_id",
99
- "environment.envive_enabled"
100
- ]
96
+ allowedFields: [],
97
+ fieldProjections: {
98
+ page_type: "context.page_type",
99
+ page_id: "context.page_id",
100
+ context_supported: "context.supported",
101
+ context_ready: "context.ready",
102
+ context_page_variant_id: "context.page_variant_id",
103
+ environment_envive_enabled: "environment.envive_enabled"
104
+ }
101
105
  },
102
106
  [EnviveMetricsEventName.WidgetTextRequest]: { gaEventName: null },
103
107
  [EnviveMetricsEventName.WidgetTextResponse]: { gaEventName: null },
@@ -106,4 +110,4 @@ const GA4_EVENT_SCHEMA = {
106
110
 
107
111
  //#endregion
108
112
  export { GA4_EVENT_SCHEMA };
109
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ga4EventSchema.js","names":["WIDGET_INTERACTION_DATA_PROJECTIONS: WidgetInteractionDataProjection","GA4_EVENT_SCHEMA: Record<EnviveMetricsEventName, GA4EventSchemaEntry>"],"sources":["../../../src/services/ga4ProjectionService/ga4EventSchema.ts"],"sourcesContent":["/**\n * GA4 Event Schema — single source of truth for Envive → GA4 projection.\n *\n * Every EnviveMetricsEventName MUST have an entry here. The Record type enforces\n * this at compile time — adding a new enum member without a schema entry will\n * cause a TypeScript error.\n *\n * ## How to update this file\n *\n * When adding or modifying event properties, decide whether they belong in `allowedFields` using these criteria:\n * - **Low cardinality** — avoid unbounded values (free text, IDs, URLs, timestamps).\n * - **Non-sensitive** — no PII, user input text, or internal implementation details.\n * - **Flat scalar** — GA4 custom dimensions are flat strings/numbers/booleans.\n *   Nested objects should NOT be added; extract specific sub-fields instead.\n * - **Snake_case key** — must already be dot-notation snake_case (e.g. `chat.user_typed`).\n * - **Analytically useful** — the field should enable meaningful filtering/segmentation\n *   for merchants in their GA4 dashboards.\n *\n * If excluded from `allowedFields`, the projection layer will automatically drop it.\n *\n * IF IN DOUBT, DON'T HESITATE TO REACH OUT TO THE ANALYTICS TEAM!!!\n */\nimport { EnviveMetricsEventName } from '../amplitudeService/eventNames';\n\n/**\n * Defines which fields to extract from `trigger.widget_interaction_data`\n * for a given `trigger.widget_interaction` value.\n *\n * Key: the value of `trigger.widget_interaction` (e.g. \"widget_collapsed\")\n * Value: mapping of { ga4FlatKey: sourceFieldInWidgetInteractionData }\n */\nexport type WidgetInteractionDataProjection = Record<string, Record<string, string>>;\n\nexport interface GA4ProjectedEventConfig {\n  gaEventName: string;\n  allowedFields: readonly string[];\n  widgetInteractionDataProjections?: WidgetInteractionDataProjection;\n}\n\nexport interface GA4ExcludedEventConfig {\n  gaEventName: null;\n}\n\nexport type GA4EventSchemaEntry = GA4ProjectedEventConfig | GA4ExcludedEventConfig;\n\nconst WIDGET_INTERACTION_DATA_PROJECTIONS: WidgetInteractionDataProjection = {\n  widget_collapsed: { trigger_collapse_source: 'collapse_source' },\n  product_card_clicked: { trigger_product_id: 'product_id' },\n  suggestion_scrolled: { trigger_suggestion_id: 'suggestion_id' },\n};\n\nexport const GA4_EVENT_SCHEMA: Record<EnviveMetricsEventName, GA4EventSchemaEntry> = {\n  [EnviveMetricsEventName.WidgetRendered]: {\n    gaEventName: 'envive_widget_rendered',\n    allowedFields: [\n      'context.page_type',\n      'context.page_id',\n      'trigger.widget',\n      'trigger.widget_role',\n      'context.surface',\n    ],\n  },\n\n  // This event is not currently implemented\n  [EnviveMetricsEventName.WidgetInteraction]: {\n    gaEventName: 'envive_widget_interaction',\n    allowedFields: [\n      'context.page_type',\n      'context.page_id',\n      'trigger.widget',\n      'trigger.widget_interaction',\n      'trigger.entity_role',\n      'context.surface',\n    ],\n    widgetInteractionDataProjections: WIDGET_INTERACTION_DATA_PROJECTIONS,\n  },\n\n  [EnviveMetricsEventName.ChatRequest]: {\n    gaEventName: 'envive_chat_request',\n    allowedFields: [\n      'page_type',\n      'page_id',\n      'trigger.widget',\n      'chat.request_type',\n      // 'chat.request_text', // This is high cardinality and potentially PII but we might want to include it later\n      'chat.user_typed',\n      'chat.suggestion_id', // not currently implemented\n      'chat.suggestion_category', // not currently implemented\n      'chat.suggestion_created_at', // not currently implemented\n      'chat.suggestion_is_answer', // not currently implemented\n      'chat.form_type', // not currently implemented\n    ],\n  },\n\n  [EnviveMetricsEventName.ChatResponse]: {\n    gaEventName: 'envive_chat_response',\n    allowedFields: [\n      'page_type',\n      'page_id',\n      'trigger.widget', // not currently implemented (maybe it should'nt be???)\n      'chat.user_typed',\n      'chat.response_time_ms',\n      'chat.product_cards_returned',\n      'chat.product_ids_returned',\n      'chat.review_cards_returned',\n    ],\n  },\n\n  [EnviveMetricsEventName.EnviveInitialized]: {\n    gaEventName: 'envive_initialized',\n    allowedFields: [\n      'environment.sales_agent_enabled',\n      'environment.envive_enabled',\n      // 'environment.search_enabled', // For now, search is not implemented. We will add it later.\n      'performance.start_time_ms',\n      'performance.initialize_time_ms',\n    ],\n  },\n\n  // This event is not currently implemented\n  [EnviveMetricsEventName.PageViewed]: {\n    gaEventName: 'envive_page_context_evaluated',\n    allowedFields: [\n      'context.page_type',\n      'context.page_id',\n      'context.supported',\n      'context.ready',\n      'context.page_variant_id',\n      'environment.envive_enabled',\n    ],\n  },\n\n  [EnviveMetricsEventName.WidgetTextRequest]: {\n    gaEventName: null,\n  },\n\n  [EnviveMetricsEventName.WidgetTextResponse]: {\n    gaEventName: null,\n  },\n\n  [EnviveMetricsEventName.WidgetTextClicked]: {\n    gaEventName: null,\n  },\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;AA6CA,MAAMA,sCAAuE;CAC3E,kBAAkB,EAAE,yBAAyB,mBAAmB;CAChE,sBAAsB,EAAE,oBAAoB,cAAc;CAC1D,qBAAqB,EAAE,uBAAuB,iBAAiB;CAChE;AAED,MAAaC,mBAAwE;EAClF,uBAAuB,iBAAiB;EACvC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GACA;GACD;EACF;EAGA,uBAAuB,oBAAoB;EAC1C,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GACA;GACA;GACD;EACD,kCAAkC;EACnC;EAEA,uBAAuB,cAAc;EACpC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GAEA;GACA;GACA;GACA;GACA;GACA;GACD;EACF;EAEA,uBAAuB,eAAe;EACrC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GACA;GACA;GACA;GACA;GACD;EACF;EAEA,uBAAuB,oBAAoB;EAC1C,aAAa;EACb,eAAe;GACb;GACA;GAEA;GACA;GACD;EACF;EAGA,uBAAuB,aAAa;EACnC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GACA;GACA;GACD;EACF;EAEA,uBAAuB,oBAAoB,EAC1C,aAAa,MACd;EAEA,uBAAuB,qBAAqB,EAC3C,aAAa,MACd;EAEA,uBAAuB,oBAAoB,EAC1C,aAAa,MACd;CACF"}
113
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ga4EventSchema.js","names":["WIDGET_INTERACTION_DATA_PROJECTIONS: WidgetInteractionDataProjection","GA4_EVENT_SCHEMA: Record<EnviveMetricsEventName, GA4EventSchemaEntry>"],"sources":["../../../src/services/ga4ProjectionService/ga4EventSchema.ts"],"sourcesContent":["/**\n * GA4 Event Schema — single source of truth for Envive → GA4 projection.\n *\n * Every EnviveMetricsEventName MUST have an entry here. The Record type enforces\n * this at compile time — adding a new enum member without a schema entry will\n * cause a TypeScript error.\n *\n * ## How to update this file\n *\n * When adding or modifying event properties, decide whether they belong in `allowedFields` using these criteria:\n * - **Low cardinality** — avoid unbounded values (free text, IDs, URLs, timestamps).\n * - **Non-sensitive** — no PII, user input text, or internal implementation details.\n * - **Flat scalar** — GA4 custom dimensions are flat strings/numbers/booleans.\n *   Nested objects should NOT be added; extract specific sub-fields instead.\n * - **Snake_case key** — must already be dot-notation snake_case (e.g. `chat.user_typed`).\n * - **Analytically useful** — the field should enable meaningful filtering/segmentation\n *   for merchants in their GA4 dashboards.\n *\n * If excluded from `allowedFields`, the projection layer will automatically drop it.\n *\n * IF IN DOUBT, DON'T HESITATE TO REACH OUT TO THE ANALYTICS TEAM!!!\n */\nimport { EnviveMetricsEventName } from '../amplitudeService/eventNames';\n\n/**\n * Defines which fields to extract from `trigger.widget_interaction_data`\n * for a given `trigger.widget_interaction` value.\n *\n * Key: the value of `trigger.widget_interaction` (e.g. \"widget_collapsed\")\n * Value: mapping of { ga4FlatKey: sourceFieldInWidgetInteractionData }\n */\nexport type WidgetInteractionDataProjection = Record<string, Record<string, string>>;\n\nexport interface GA4ProjectedEventConfig {\n  gaEventName: string;\n  allowedFields: readonly string[];\n  /**\n   * When present, used instead of `allowedFields` to both filter and rename fields.\n   * Key: the desired GA4 parameter name. Value: the source dot-notation key in eventProps.\n   */\n  fieldProjections?: Record<string, string>;\n  widgetInteractionDataProjections?: WidgetInteractionDataProjection;\n}\n\nexport interface GA4ExcludedEventConfig {\n  gaEventName: null;\n}\n\nexport type GA4EventSchemaEntry = GA4ProjectedEventConfig | GA4ExcludedEventConfig;\n\nconst WIDGET_INTERACTION_DATA_PROJECTIONS: WidgetInteractionDataProjection = {\n  widget_collapsed: { interaction_collapse_source: 'widget_collapsed.collapse_source' },\n  product_card_clicked: { interaction_product_id: 'product_card_clicked.product_id' },\n  suggestion_scrolled: { interaction_suggestion_id: 'suggestion_scrolled.suggestion_id' },\n};\n\nexport const GA4_EVENT_SCHEMA: Record<EnviveMetricsEventName, GA4EventSchemaEntry> = {\n  [EnviveMetricsEventName.WidgetRendered]: {\n    gaEventName: 'envive_widget_rendered',\n    allowedFields: [],\n    fieldProjections: {\n      page_type: 'context.page_type',\n      page_id: 'context.page_id',\n      surface: 'context.surface',\n      widget: 'trigger.widget',\n      widget_role: 'trigger.widget_role',\n    },\n  },\n\n  [EnviveMetricsEventName.WidgetInteraction]: {\n    gaEventName: 'envive_widget_interaction',\n    allowedFields: [],\n    fieldProjections: {\n      page_type: 'context.page_type',\n      page_id: 'context.page_id',\n      surface: 'context.surface',\n      widget: 'trigger.widget',\n      interaction_type: 'trigger.widget_interaction',\n      interaction_class: 'trigger.interaction_class',\n      widget_role: 'trigger.widget_role',\n    },\n    widgetInteractionDataProjections: WIDGET_INTERACTION_DATA_PROJECTIONS,\n  },\n\n  [EnviveMetricsEventName.ChatRequest]: {\n    gaEventName: 'envive_chat_request',\n    allowedFields: [\n      'page_type',\n      'page_id',\n      'trigger.widget',\n      'chat.request_type',\n      // 'chat.request_text', // This is high cardinality and potentially PII but we might want to include it later\n      'chat.user_typed',\n      'chat.suggestion_id', // not currently implemented\n      'chat.suggestion_category', // not currently implemented\n      'chat.suggestion_created_at', // not currently implemented\n      'chat.suggestion_is_answer', // not currently implemented\n      'chat.form_type', // not currently implemented\n    ],\n  },\n\n  [EnviveMetricsEventName.ChatResponse]: {\n    gaEventName: 'envive_chat_response',\n    allowedFields: [\n      'page_type',\n      'page_id',\n      'trigger.widget', // not currently implemented (maybe it should'nt be???)\n      'chat.user_typed',\n      'chat.response_time_ms',\n      'chat.product_cards_returned',\n      'chat.product_ids_returned',\n      'chat.review_cards_returned',\n    ],\n  },\n\n  [EnviveMetricsEventName.EnviveInitialized]: {\n    gaEventName: 'envive_initialized',\n    allowedFields: [\n      'environment.sales_agent_enabled',\n      'environment.envive_enabled',\n      // 'environment.search_enabled', // For now, search is not implemented. We will add it later.\n      'performance.start_time_ms',\n      'performance.initialize_time_ms',\n    ],\n  },\n\n  // This event is not currently implemented\n  [EnviveMetricsEventName.PageViewed]: {\n    gaEventName: 'envive_page_context_evaluated',\n    allowedFields: [],\n    fieldProjections: {\n      page_type: 'context.page_type',\n      page_id: 'context.page_id',\n      context_supported: 'context.supported',\n      context_ready: 'context.ready',\n      context_page_variant_id: 'context.page_variant_id',\n      environment_envive_enabled: 'environment.envive_enabled',\n    },\n  },\n\n  [EnviveMetricsEventName.WidgetTextRequest]: {\n    gaEventName: null,\n  },\n\n  [EnviveMetricsEventName.WidgetTextResponse]: {\n    gaEventName: null,\n  },\n\n  [EnviveMetricsEventName.WidgetTextClicked]: {\n    gaEventName: null,\n  },\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;AAkDA,MAAMA,sCAAuE;CAC3E,kBAAkB,EAAE,6BAA6B,oCAAoC;CACrF,sBAAsB,EAAE,wBAAwB,mCAAmC;CACnF,qBAAqB,EAAE,2BAA2B,qCAAqC;CACxF;AAED,MAAaC,mBAAwE;EAClF,uBAAuB,iBAAiB;EACvC,aAAa;EACb,eAAe,EAAE;EACjB,kBAAkB;GAChB,WAAW;GACX,SAAS;GACT,SAAS;GACT,QAAQ;GACR,aAAa;GACd;EACF;EAEA,uBAAuB,oBAAoB;EAC1C,aAAa;EACb,eAAe,EAAE;EACjB,kBAAkB;GAChB,WAAW;GACX,SAAS;GACT,SAAS;GACT,QAAQ;GACR,kBAAkB;GAClB,mBAAmB;GACnB,aAAa;GACd;EACD,kCAAkC;EACnC;EAEA,uBAAuB,cAAc;EACpC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GAEA;GACA;GACA;GACA;GACA;GACA;GACD;EACF;EAEA,uBAAuB,eAAe;EACrC,aAAa;EACb,eAAe;GACb;GACA;GACA;GACA;GACA;GACA;GACA;GACA;GACD;EACF;EAEA,uBAAuB,oBAAoB;EAC1C,aAAa;EACb,eAAe;GACb;GACA;GAEA;GACA;GACD;EACF;EAGA,uBAAuB,aAAa;EACnC,aAAa;EACb,eAAe,EAAE;EACjB,kBAAkB;GAChB,WAAW;GACX,SAAS;GACT,mBAAmB;GACnB,eAAe;GACf,yBAAyB;GACzB,4BAA4B;GAC7B;EACF;EAEA,uBAAuB,oBAAoB,EAC1C,aAAa,MACd;EAEA,uBAAuB,qBAAqB,EAC3C,aAAa,MACd;EAEA,uBAAuB,oBAAoB,EAC1C,aAAa,MACd;CACF"}
@@ -8,18 +8,39 @@ const filterToSchema = (eventProps, allowedFields) => {
8
8
  for (const field of allowedFields) if (field in eventProps) result[field] = eventProps[field];
9
9
  return result;
10
10
  };
11
+ const flattenOneLevel = (obj) => {
12
+ const result = {};
13
+ for (const [key, value] of Object.entries(obj)) if (!key.includes(".") && value !== null && typeof value === "object" && !Array.isArray(value)) for (const [subKey, subValue] of Object.entries(value)) result[`${key}.${subKey}`] = subValue;
14
+ else result[key] = value;
15
+ return result;
16
+ };
17
+ const filterWithProjections = (eventProps, projections) => {
18
+ const result = {};
19
+ for (const [ga4Key, sourceKey] of Object.entries(projections)) if (sourceKey in eventProps) result[ga4Key] = eventProps[sourceKey];
20
+ return result;
21
+ };
11
22
  const flattenDotKeys = (obj) => {
12
23
  const result = {};
13
24
  for (const [key, value] of Object.entries(obj)) result[key.replace(/\./g, "_")] = value;
14
25
  return result;
15
26
  };
16
27
  const sanitizePageId = (filtered) => {
17
- const pageType = filtered["context.page_type"];
28
+ const pageType = filtered["page_type"] ?? filtered["context.page_type"];
18
29
  if (pageType === "pdp" || pageType === "plp") return filtered;
19
30
  const rest = { ...filtered };
31
+ delete rest["page_id"];
20
32
  delete rest["context.page_id"];
21
33
  return rest;
22
34
  };
35
+ const getNestedValue = (obj, path) => {
36
+ const parts = path.split(".");
37
+ let current = obj;
38
+ for (const part of parts) {
39
+ if (current === null || typeof current !== "object") return void 0;
40
+ current = current[part];
41
+ }
42
+ return current;
43
+ };
23
44
  const projectWidgetInteractionData = (eventProps, config) => {
24
45
  if (!config.widgetInteractionDataProjections) return {};
25
46
  const interaction = eventProps["trigger.widget_interaction"];
@@ -30,7 +51,10 @@ const projectWidgetInteractionData = (eventProps, config) => {
30
51
  if (interactionData === null || interactionData === void 0 || typeof interactionData !== "object") return {};
31
52
  const data = interactionData;
32
53
  const result = {};
33
- for (const [gaKey, sourceField] of Object.entries(projectionMap)) if (sourceField in data) result[gaKey] = data[sourceField];
54
+ for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {
55
+ const value = getNestedValue(data, sourcePath);
56
+ if (value !== void 0) result[gaKey] = value;
57
+ }
34
58
  return result;
35
59
  };
36
60
  const truncateString = (value) => {
@@ -50,8 +74,10 @@ const projectToGA4 = (eventName, eventProps) => {
50
74
  const schemaEntry = require_ga4EventSchema.GA4_EVENT_SCHEMA[eventName];
51
75
  if (schemaEntry.gaEventName === null) return;
52
76
  const config = schemaEntry;
53
- const props = eventProps ?? {};
54
- let filtered = filterToSchema(props, config.allowedFields);
77
+ const props = flattenOneLevel(eventProps ?? {});
78
+ let filtered;
79
+ if (config.fieldProjections) filtered = filterWithProjections(props, config.fieldProjections);
80
+ else filtered = filterToSchema(props, config.allowedFields);
55
81
  filtered = sanitizePageId(filtered);
56
82
  const interactionFields = projectWidgetInteractionData(props, config);
57
83
  const truncatedParams = truncateValues({
@@ -69,4 +95,4 @@ const projectToGA4 = (eventName, eventProps) => {
69
95
 
70
96
  //#endregion
71
97
  exports.projectToGA4 = projectToGA4;
72
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2E0UHJvamVjdGlvblNlcnZpY2UuY2pzIiwibmFtZXMiOlsiTG9nZ2VyIiwicmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiIsIkdBNF9FVkVOVF9TQ0hFTUEiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VydmljZXMvZ2E0UHJvamVjdGlvblNlcnZpY2UvZ2E0UHJvamVjdGlvblNlcnZpY2UudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IExvZ2dlciBmcm9tICdzcmMvYXBwbGljYXRpb24vbG9nZ2luZy9sb2dnZXInO1xuaW1wb3J0IHsgRW52aXZlTWV0cmljc0V2ZW50TmFtZSB9IGZyb20gJy4uL2FtcGxpdHVkZVNlcnZpY2UvZXZlbnROYW1lcyc7XG5pbXBvcnQgeyBHQTRQcm9qZWN0ZWRFdmVudENvbmZpZywgR0E0X0VWRU5UX1NDSEVNQSB9IGZyb20gJy4vZ2E0RXZlbnRTY2hlbWEnO1xuXG5jb25zdCBsb2dnZXIgPSBuZXcgTG9nZ2VyKCdnYTRQcm9qZWN0aW9uU2VydmljZScpO1xuXG5jb25zdCBmaWx0ZXJUb1NjaGVtYSA9IChcbiAgZXZlbnRQcm9wczogUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gIGFsbG93ZWRGaWVsZHM6IHJlYWRvbmx5IHN0cmluZ1tdLFxuKTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPT4ge1xuICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG4gIGZvciAoY29uc3QgZmllbGQgb2YgYWxsb3dlZEZpZWxkcykge1xuICAgIGlmIChmaWVsZCBpbiBldmVudFByb3BzKSB7XG4gICAgICByZXN1bHRbZmllbGRdID0gZXZlbnRQcm9wc1tmaWVsZF07XG4gICAgfVxuICB9XG4gIHJldHVybiByZXN1bHQ7XG59O1xuXG4vLyBcImNvbnRleHQucGFnZV90eXBlXCIg4oaSIFwiY29udGV4dF9wYWdlX3R5cGVcIlxuY29uc3QgZmxhdHRlbkRvdEtleXMgPSAob2JqOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhvYmopKSB7XG4gICAgcmVzdWx0W2tleS5yZXBsYWNlKC9cXC4vZywgJ18nKV0gPSB2YWx1ZTtcbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuLy8gT21pdCBjb250ZXh0LnBhZ2VfaWQgZm9yIG5vbi1wZHAvcGxwIHBhZ2UgdHlwZXMuIFRoZSBjdXJyZW50IGltcGxlbWVudGF0aW9uIGZvciBjb250ZXh0LnBhZ2VfaWQgaXM6XG4vLyBQRFA6IHByb2R1Y3RfaWRcbi8vIFBMUDogcGxwX2lkXG4vLyBTZWFyY2g6IHNlYXJjaCBxdWVyeVxuLy8gT3RoZXI6IHBhZ2UgdXJsXG4vLyBXZSB3YW50IHRvIG9taXQgYWxsIGJ1dCBwZHAgYW5kIHBscCBwYWdlIHR5cGVzIHRvIHByb3ZpZGUgYSBjbGVhciwgY29uc2lzdGVudCBpbnRlcmZhY2UgZm9yIG1lcmNoYW50cy5cbmNvbnN0IHNhbml0aXplUGFnZUlkID0gKGZpbHRlcmVkOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcGFnZVR5cGUgPSBmaWx0ZXJlZFsnY29udGV4dC5wYWdlX3R5cGUnXTtcbiAgaWYgKHBhZ2VUeXBlID09PSAncGRwJyB8fCBwYWdlVHlwZSA9PT0gJ3BscCcpIHtcbiAgICByZXR1cm4gZmlsdGVyZWQ7XG4gIH1cbiAgY29uc3QgcmVzdCA9IHsgLi4uZmlsdGVyZWQgfTtcbiAgZGVsZXRlIHJlc3RbJ2NvbnRleHQucGFnZV9pZCddO1xuICByZXR1cm4gcmVzdDtcbn07XG5cbi8vIEV4dHJhY3Qgd2hpdGVsaXN0ZWQgc3ViLWZpZWxkcyBmcm9tIHRyaWdnZXIud2lkZ2V0X2ludGVyYWN0aW9uX2RhdGFcbmNvbnN0IHByb2plY3RXaWRnZXRJbnRlcmFjdGlvbkRhdGEgPSAoXG4gIGV2ZW50UHJvcHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuICBjb25maWc6IEdBNFByb2plY3RlZEV2ZW50Q29uZmlnLFxuKTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPT4ge1xuICBpZiAoIWNvbmZpZy53aWRnZXRJbnRlcmFjdGlvbkRhdGFQcm9qZWN0aW9ucykge1xuICAgIHJldHVybiB7fTtcbiAgfVxuXG4gIGNvbnN0IGludGVyYWN0aW9uID0gZXZlbnRQcm9wc1sndHJpZ2dlci53aWRnZXRfaW50ZXJhY3Rpb24nXTtcbiAgaWYgKHR5cGVvZiBpbnRlcmFjdGlvbiAhPT0gJ3N0cmluZycpIHtcbiAgICByZXR1cm4ge307XG4gIH1cblxuICBjb25zdCBwcm9qZWN0aW9uTWFwID0gY29uZmlnLndpZGdldEludGVyYWN0aW9uRGF0YVByb2plY3Rpb25zW2ludGVyYWN0aW9uXTtcbiAgaWYgKCFwcm9qZWN0aW9uTWFwKSB7XG4gICAgcmV0dXJuIHt9O1xuICB9XG5cbiAgY29uc3QgaW50ZXJhY3Rpb25EYXRhID0gZXZlbnRQcm9wc1sndHJpZ2dlci53aWRnZXRfaW50ZXJhY3Rpb25fZGF0YSddO1xuICBpZiAoXG4gICAgaW50ZXJhY3Rpb25EYXRhID09PSBudWxsIHx8XG4gICAgaW50ZXJhY3Rpb25EYXRhID09PSB1bmRlZmluZWQgfHxcbiAgICB0eXBlb2YgaW50ZXJhY3Rpb25EYXRhICE9PSAnb2JqZWN0J1xuICApIHtcbiAgICByZXR1cm4ge307XG4gIH1cblxuICBjb25zdCBkYXRhID0gaW50ZXJhY3Rpb25EYXRhIGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG4gIGZvciAoY29uc3QgW2dhS2V5LCBzb3VyY2VGaWVsZF0gb2YgT2JqZWN0LmVudHJpZXMocHJvamVjdGlvbk1hcCkpIHtcbiAgICBpZiAoc291cmNlRmllbGQgaW4gZGF0YSkge1xuICAgICAgcmVzdWx0W2dhS2V5XSA9IGRhdGFbc291cmNlRmllbGRdO1xuICAgIH1cbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuY29uc3QgdHJ1bmNhdGVTdHJpbmcgPSAodmFsdWU6IHVua25vd24pOiB1bmtub3duID0+IHtcbiAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ3N0cmluZycgJiYgdmFsdWUubGVuZ3RoID4gMTAwKSB7XG4gICAgcmV0dXJuIGAke3ZhbHVlLnN1YnN0cmluZygwLCA5Nyl9Li4uYDtcbiAgfVxuICByZXR1cm4gdmFsdWU7XG59O1xuXG5jb25zdCB0cnVuY2F0ZVZhbHVlcyA9IChvYmo6IFJlY29yZDxzdHJpbmcsIHVua25vd24+KTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPT4ge1xuICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge307XG4gIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKG9iaikpIHtcbiAgICByZXN1bHRba2V5XSA9IHRydW5jYXRlU3RyaW5nKHZhbHVlKTtcbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuY29uc3QgcHVzaFRvRGF0YUxheWVyID0gKGdhRXZlbnQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+KTogdm9pZCA9PiB7XG4gIGlmICh0eXBlb2Ygd2luZG93ICE9PSAndW5kZWZpbmVkJyAmJiB3aW5kb3cuZGF0YUxheWVyKSB7XG4gICAgd2luZG93LmRhdGFMYXllci5wdXNoKGdhRXZlbnQpO1xuICB9XG59O1xuXG5leHBvcnQgY29uc3QgcHJvamVjdFRvR0E0ID0gKFxuICBldmVudE5hbWU6IEVudml2ZU1ldHJpY3NFdmVudE5hbWUsXG4gIGV2ZW50UHJvcHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbik6IHZvaWQgPT4ge1xuICB0cnkge1xuICAgIGNvbnN0IHNjaGVtYUVudHJ5ID0gR0E0X0VWRU5UX1NDSEVNQVtldmVudE5hbWVdO1xuXG4gICAgaWYgKHNjaGVtYUVudHJ5LmdhRXZlbnROYW1lID09PSBudWxsKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgY29uZmlnID0gc2NoZW1hRW50cnk7XG4gICAgY29uc3QgcHJvcHMgPSBldmVudFByb3BzID8/IHt9O1xuXG4gICAgbGV0IGZpbHRlcmVkID0gZmlsdGVyVG9TY2hlbWEocHJvcHMsIGNvbmZpZy5hbGxvd2VkRmllbGRzKTtcbiAgICBmaWx0ZXJlZCA9IHNhbml0aXplUGFnZUlkKGZpbHRlcmVkKTtcblxuICAgIGNvbnN0IGludGVyYWN0aW9uRmllbGRzID0gcHJvamVjdFdpZGdldEludGVyYWN0aW9uRGF0YShwcm9wcywgY29uZmlnKTtcblxuICAgIGNvbnN0IGZsYXRQYXJhbXMgPSB7XG4gICAgICAuLi5mbGF0dGVuRG90S2V5cyhmaWx0ZXJlZCksXG4gICAgICAuLi5pbnRlcmFjdGlvbkZpZWxkcyxcbiAgICB9O1xuXG4gICAgY29uc3QgdHJ1bmNhdGVkUGFyYW1zID0gdHJ1bmNhdGVWYWx1ZXMoZmxhdFBhcmFtcyk7XG5cbiAgICBwdXNoVG9EYXRhTGF5ZXIoe1xuICAgICAgZXZlbnQ6IGNvbmZpZy5nYUV2ZW50TmFtZSxcbiAgICAgIC4uLnRydW5jYXRlZFBhcmFtcyxcbiAgICB9KTtcbiAgfSBjYXRjaCAoZXJyKSB7XG4gICAgbG9nZ2VyLmxvZ0Vycm9yKCdFcnJvciBwcm9qZWN0aW5nIGV2ZW50IHRvIEdBNCcsIGVyciwge1xuICAgICAgZXZlbnROYW1lLFxuICAgIH0pO1xuICB9XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7O0FBSUEsTUFBTSxTQUFTLElBQUlBLHVCQUFPLHVCQUF1QjtBQUVqRCxNQUFNLGtCQUNKLFlBQ0Esa0JBQzRCO0NBQzVCLE1BQU1DLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLFNBQVMsY0FDbEIsS0FBSSxTQUFTLFdBQ1gsUUFBTyxTQUFTLFdBQVc7QUFHL0IsUUFBTzs7QUFJVCxNQUFNLGtCQUFrQixRQUEwRDtDQUNoRixNQUFNQSxTQUFrQyxFQUFFO0FBQzFDLE1BQUssTUFBTSxDQUFDLEtBQUssVUFBVSxPQUFPLFFBQVEsSUFBSSxDQUM1QyxRQUFPLElBQUksUUFBUSxPQUFPLElBQUksSUFBSTtBQUVwQyxRQUFPOztBQVNULE1BQU0sa0JBQWtCLGFBQStEO0NBQ3JGLE1BQU0sV0FBVyxTQUFTO0FBQzFCLEtBQUksYUFBYSxTQUFTLGFBQWEsTUFDckMsUUFBTztDQUVULE1BQU0sT0FBTyxFQUFFLEdBQUcsVUFBVTtBQUM1QixRQUFPLEtBQUs7QUFDWixRQUFPOztBQUlULE1BQU0sZ0NBQ0osWUFDQSxXQUM0QjtBQUM1QixLQUFJLENBQUMsT0FBTyxpQ0FDVixRQUFPLEVBQUU7Q0FHWCxNQUFNLGNBQWMsV0FBVztBQUMvQixLQUFJLE9BQU8sZ0JBQWdCLFNBQ3pCLFFBQU8sRUFBRTtDQUdYLE1BQU0sZ0JBQWdCLE9BQU8saUNBQWlDO0FBQzlELEtBQUksQ0FBQyxjQUNILFFBQU8sRUFBRTtDQUdYLE1BQU0sa0JBQWtCLFdBQVc7QUFDbkMsS0FDRSxvQkFBb0IsUUFDcEIsb0JBQW9CLFVBQ3BCLE9BQU8sb0JBQW9CLFNBRTNCLFFBQU8sRUFBRTtDQUdYLE1BQU0sT0FBTztDQUNiLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsT0FBTyxnQkFBZ0IsT0FBTyxRQUFRLGNBQWMsQ0FDOUQsS0FBSSxlQUFlLEtBQ2pCLFFBQU8sU0FBUyxLQUFLO0FBR3pCLFFBQU87O0FBR1QsTUFBTSxrQkFBa0IsVUFBNEI7QUFDbEQsS0FBSSxPQUFPLFVBQVUsWUFBWSxNQUFNLFNBQVMsSUFDOUMsUUFBTyxHQUFHLE1BQU0sVUFBVSxHQUFHLEdBQUcsQ0FBQztBQUVuQyxRQUFPOztBQUdULE1BQU0sa0JBQWtCLFFBQTBEO0NBQ2hGLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsS0FBSyxVQUFVLE9BQU8sUUFBUSxJQUFJLENBQzVDLFFBQU8sT0FBTyxlQUFlLE1BQU07QUFFckMsUUFBTzs7QUFHVCxNQUFNLG1CQUFtQixZQUEyQztBQUNsRSxLQUFJLE9BQU8sV0FBVyxlQUFlLE9BQU8sVUFDMUMsUUFBTyxVQUFVLEtBQUssUUFBUTs7QUFJbEMsTUFBYSxnQkFDWCxXQUNBLGVBQ1M7QUFDVCxLQUFJO0VBQ0YsTUFBTSxjQUFjQyx3Q0FBaUI7QUFFckMsTUFBSSxZQUFZLGdCQUFnQixLQUM5QjtFQUdGLE1BQU0sU0FBUztFQUNmLE1BQU0sUUFBUSxjQUFjLEVBQUU7RUFFOUIsSUFBSSxXQUFXLGVBQWUsT0FBTyxPQUFPLGNBQWM7QUFDMUQsYUFBVyxlQUFlLFNBQVM7RUFFbkMsTUFBTSxvQkFBb0IsNkJBQTZCLE9BQU8sT0FBTztFQU9yRSxNQUFNLGtCQUFrQixlQUxMO0dBQ2pCLEdBQUcsZUFBZSxTQUFTO0dBQzNCLEdBQUc7R0FDSixDQUVpRDtBQUVsRCxrQkFBZ0I7R0FDZCxPQUFPLE9BQU87R0FDZCxHQUFHO0dBQ0osQ0FBQztVQUNLLEtBQUs7QUFDWixTQUFPLFNBQVMsaUNBQWlDLEtBQUssRUFDcEQsV0FDRCxDQUFDIn0=
98
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ga4ProjectionService.cjs","names":["Logger","result: Record<string, unknown>","current: unknown","GA4_EVENT_SCHEMA","filtered: Record<string, unknown>"],"sources":["../../../src/services/ga4ProjectionService/ga4ProjectionService.ts"],"sourcesContent":["import Logger from 'src/application/logging/logger';\nimport { EnviveMetricsEventName } from '../amplitudeService/eventNames';\nimport { GA4ProjectedEventConfig, GA4_EVENT_SCHEMA } from './ga4EventSchema';\n\nconst logger = new Logger('ga4ProjectionService');\n\nconst filterToSchema = (\n  eventProps: Record<string, unknown>,\n  allowedFields: readonly string[],\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const field of allowedFields) {\n    if (field in eventProps) {\n      result[field] = eventProps[field];\n    }\n  }\n  return result;\n};\n\n// Flatten only true top-level nested objects (keys with no dots) one level deep.\n// Keys that already contain dots are left intact so that e.g.\n// `trigger.widget_interaction_data: { ... }` is preserved for downstream extraction.\nconst flattenOneLevel = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    if (\n      !key.includes('.') &&\n      value !== null &&\n      typeof value === 'object' &&\n      !Array.isArray(value)\n    ) {\n      for (const [subKey, subValue] of Object.entries(value as Record<string, unknown>)) {\n        result[`${key}.${subKey}`] = subValue;\n      }\n    } else {\n      result[key] = value;\n    }\n  }\n  return result;\n};\n\n// Apply a GA4 projection map { ga4Key: sourceKey } to extract and rename fields.\nconst filterWithProjections = (\n  eventProps: Record<string, unknown>,\n  projections: Record<string, string>,\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [ga4Key, sourceKey] of Object.entries(projections)) {\n    if (sourceKey in eventProps) {\n      result[ga4Key] = eventProps[sourceKey];\n    }\n  }\n  return result;\n};\n\n// \"context.page_type\" → \"context_page_type\"\nconst flattenDotKeys = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key.replace(/\\./g, '_')] = value;\n  }\n  return result;\n};\n\n// Omit context.page_id for non-pdp/plp page types. The current implementation for context.page_id is:\n// PDP: product_id\n// PLP: plp_id\n// Search: search query\n// Other: page url\n// We want to omit all but pdp and plp page types to provide a clear, consistent interface for merchants.\n// Handles both dot-notation keys (allowedFields path) and renamed GA4 keys (fieldProjections path).\nconst sanitizePageId = (filtered: Record<string, unknown>): Record<string, unknown> => {\n  const pageType = filtered['page_type'] ?? filtered['context.page_type'];\n  if (pageType === 'pdp' || pageType === 'plp') {\n    return filtered;\n  }\n  const rest = { ...filtered };\n  delete rest['page_id'];\n  delete rest['context.page_id'];\n  return rest;\n};\n\nconst getNestedValue = (obj: Record<string, unknown>, path: string): unknown => {\n  const parts = path.split('.');\n  let current: unknown = obj;\n  for (const part of parts) {\n    if (current === null || typeof current !== 'object') return undefined;\n    current = (current as Record<string, unknown>)[part];\n  }\n  return current;\n};\n\n// Extract whitelisted sub-fields from trigger.widget_interaction_data\nconst projectWidgetInteractionData = (\n  eventProps: Record<string, unknown>,\n  config: GA4ProjectedEventConfig,\n): Record<string, unknown> => {\n  if (!config.widgetInteractionDataProjections) {\n    return {};\n  }\n\n  const interaction = eventProps['trigger.widget_interaction'];\n  if (typeof interaction !== 'string') {\n    return {};\n  }\n\n  const projectionMap = config.widgetInteractionDataProjections[interaction];\n  if (!projectionMap) {\n    return {};\n  }\n\n  const interactionData = eventProps['trigger.widget_interaction_data'];\n  if (\n    interactionData === null ||\n    interactionData === undefined ||\n    typeof interactionData !== 'object'\n  ) {\n    return {};\n  }\n\n  const data = interactionData as Record<string, unknown>;\n  const result: Record<string, unknown> = {};\n  for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {\n    const value = getNestedValue(data, sourcePath);\n    if (value !== undefined) {\n      result[gaKey] = value;\n    }\n  }\n  return result;\n};\n\nconst truncateString = (value: unknown): unknown => {\n  if (typeof value === 'string' && value.length > 100) {\n    return `${value.substring(0, 97)}...`;\n  }\n  return value;\n};\n\nconst truncateValues = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key] = truncateString(value);\n  }\n  return result;\n};\n\nconst pushToDataLayer = (gaEvent: Record<string, unknown>): void => {\n  if (typeof window !== 'undefined' && window.dataLayer) {\n    window.dataLayer.push(gaEvent);\n  }\n};\n\nexport const projectToGA4 = (\n  eventName: EnviveMetricsEventName,\n  eventProps?: Record<string, unknown>,\n): void => {\n  try {\n    const schemaEntry = GA4_EVENT_SCHEMA[eventName];\n\n    if (schemaEntry.gaEventName === null) {\n      return;\n    }\n\n    const config = schemaEntry;\n    const props = flattenOneLevel(eventProps ?? {});\n\n    let filtered: Record<string, unknown>;\n    if (config.fieldProjections) {\n      filtered = filterWithProjections(props, config.fieldProjections);\n    } else {\n      filtered = filterToSchema(props, config.allowedFields);\n    }\n    filtered = sanitizePageId(filtered);\n\n    const interactionFields = projectWidgetInteractionData(props, config);\n\n    const flatParams = {\n      ...flattenDotKeys(filtered),\n      ...interactionFields,\n    };\n\n    const truncatedParams = truncateValues(flatParams);\n\n    pushToDataLayer({\n      event: config.gaEventName,\n      ...truncatedParams,\n    });\n  } catch (err) {\n    logger.logError('Error projecting event to GA4', err, {\n      eventName,\n    });\n  }\n};\n"],"mappings":"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"}
@@ -8,18 +8,39 @@ const filterToSchema = (eventProps, allowedFields) => {
8
8
  for (const field of allowedFields) if (field in eventProps) result[field] = eventProps[field];
9
9
  return result;
10
10
  };
11
+ const flattenOneLevel = (obj) => {
12
+ const result = {};
13
+ for (const [key, value] of Object.entries(obj)) if (!key.includes(".") && value !== null && typeof value === "object" && !Array.isArray(value)) for (const [subKey, subValue] of Object.entries(value)) result[`${key}.${subKey}`] = subValue;
14
+ else result[key] = value;
15
+ return result;
16
+ };
17
+ const filterWithProjections = (eventProps, projections) => {
18
+ const result = {};
19
+ for (const [ga4Key, sourceKey] of Object.entries(projections)) if (sourceKey in eventProps) result[ga4Key] = eventProps[sourceKey];
20
+ return result;
21
+ };
11
22
  const flattenDotKeys = (obj) => {
12
23
  const result = {};
13
24
  for (const [key, value] of Object.entries(obj)) result[key.replace(/\./g, "_")] = value;
14
25
  return result;
15
26
  };
16
27
  const sanitizePageId = (filtered) => {
17
- const pageType = filtered["context.page_type"];
28
+ const pageType = filtered["page_type"] ?? filtered["context.page_type"];
18
29
  if (pageType === "pdp" || pageType === "plp") return filtered;
19
30
  const rest = { ...filtered };
31
+ delete rest["page_id"];
20
32
  delete rest["context.page_id"];
21
33
  return rest;
22
34
  };
35
+ const getNestedValue = (obj, path) => {
36
+ const parts = path.split(".");
37
+ let current = obj;
38
+ for (const part of parts) {
39
+ if (current === null || typeof current !== "object") return void 0;
40
+ current = current[part];
41
+ }
42
+ return current;
43
+ };
23
44
  const projectWidgetInteractionData = (eventProps, config) => {
24
45
  if (!config.widgetInteractionDataProjections) return {};
25
46
  const interaction = eventProps["trigger.widget_interaction"];
@@ -30,7 +51,10 @@ const projectWidgetInteractionData = (eventProps, config) => {
30
51
  if (interactionData === null || interactionData === void 0 || typeof interactionData !== "object") return {};
31
52
  const data = interactionData;
32
53
  const result = {};
33
- for (const [gaKey, sourceField] of Object.entries(projectionMap)) if (sourceField in data) result[gaKey] = data[sourceField];
54
+ for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {
55
+ const value = getNestedValue(data, sourcePath);
56
+ if (value !== void 0) result[gaKey] = value;
57
+ }
34
58
  return result;
35
59
  };
36
60
  const truncateString = (value) => {
@@ -50,8 +74,10 @@ const projectToGA4 = (eventName, eventProps) => {
50
74
  const schemaEntry = GA4_EVENT_SCHEMA[eventName];
51
75
  if (schemaEntry.gaEventName === null) return;
52
76
  const config = schemaEntry;
53
- const props = eventProps ?? {};
54
- let filtered = filterToSchema(props, config.allowedFields);
77
+ const props = flattenOneLevel(eventProps ?? {});
78
+ let filtered;
79
+ if (config.fieldProjections) filtered = filterWithProjections(props, config.fieldProjections);
80
+ else filtered = filterToSchema(props, config.allowedFields);
55
81
  filtered = sanitizePageId(filtered);
56
82
  const interactionFields = projectWidgetInteractionData(props, config);
57
83
  const truncatedParams = truncateValues({
@@ -69,4 +95,4 @@ const projectToGA4 = (eventName, eventProps) => {
69
95
 
70
96
  //#endregion
71
97
  export { projectToGA4 };
72
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2E0UHJvamVjdGlvblNlcnZpY2UuanMiLCJuYW1lcyI6WyJMb2dnZXIiLCJyZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+Il0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3NlcnZpY2VzL2dhNFByb2plY3Rpb25TZXJ2aWNlL2dhNFByb2plY3Rpb25TZXJ2aWNlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBMb2dnZXIgZnJvbSAnc3JjL2FwcGxpY2F0aW9uL2xvZ2dpbmcvbG9nZ2VyJztcbmltcG9ydCB7IEVudml2ZU1ldHJpY3NFdmVudE5hbWUgfSBmcm9tICcuLi9hbXBsaXR1ZGVTZXJ2aWNlL2V2ZW50TmFtZXMnO1xuaW1wb3J0IHsgR0E0UHJvamVjdGVkRXZlbnRDb25maWcsIEdBNF9FVkVOVF9TQ0hFTUEgfSBmcm9tICcuL2dhNEV2ZW50U2NoZW1hJztcblxuY29uc3QgbG9nZ2VyID0gbmV3IExvZ2dlcignZ2E0UHJvamVjdGlvblNlcnZpY2UnKTtcblxuY29uc3QgZmlsdGVyVG9TY2hlbWEgPSAoXG4gIGV2ZW50UHJvcHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuICBhbGxvd2VkRmllbGRzOiByZWFkb25seSBzdHJpbmdbXSxcbik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IGZpZWxkIG9mIGFsbG93ZWRGaWVsZHMpIHtcbiAgICBpZiAoZmllbGQgaW4gZXZlbnRQcm9wcykge1xuICAgICAgcmVzdWx0W2ZpZWxkXSA9IGV2ZW50UHJvcHNbZmllbGRdO1xuICAgIH1cbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuLy8gXCJjb250ZXh0LnBhZ2VfdHlwZVwiIOKGkiBcImNvbnRleHRfcGFnZV90eXBlXCJcbmNvbnN0IGZsYXR0ZW5Eb3RLZXlzID0gKG9iajogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9PiB7XG4gIGNvbnN0IHJlc3VsdDogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSB7fTtcbiAgZm9yIChjb25zdCBba2V5LCB2YWx1ZV0gb2YgT2JqZWN0LmVudHJpZXMob2JqKSkge1xuICAgIHJlc3VsdFtrZXkucmVwbGFjZSgvXFwuL2csICdfJyldID0gdmFsdWU7XG4gIH1cbiAgcmV0dXJuIHJlc3VsdDtcbn07XG5cbi8vIE9taXQgY29udGV4dC5wYWdlX2lkIGZvciBub24tcGRwL3BscCBwYWdlIHR5cGVzLiBUaGUgY3VycmVudCBpbXBsZW1lbnRhdGlvbiBmb3IgY29udGV4dC5wYWdlX2lkIGlzOlxuLy8gUERQOiBwcm9kdWN0X2lkXG4vLyBQTFA6IHBscF9pZFxuLy8gU2VhcmNoOiBzZWFyY2ggcXVlcnlcbi8vIE90aGVyOiBwYWdlIHVybFxuLy8gV2Ugd2FudCB0byBvbWl0IGFsbCBidXQgcGRwIGFuZCBwbHAgcGFnZSB0eXBlcyB0byBwcm92aWRlIGEgY2xlYXIsIGNvbnNpc3RlbnQgaW50ZXJmYWNlIGZvciBtZXJjaGFudHMuXG5jb25zdCBzYW5pdGl6ZVBhZ2VJZCA9IChmaWx0ZXJlZDogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9PiB7XG4gIGNvbnN0IHBhZ2VUeXBlID0gZmlsdGVyZWRbJ2NvbnRleHQucGFnZV90eXBlJ107XG4gIGlmIChwYWdlVHlwZSA9PT0gJ3BkcCcgfHwgcGFnZVR5cGUgPT09ICdwbHAnKSB7XG4gICAgcmV0dXJuIGZpbHRlcmVkO1xuICB9XG4gIGNvbnN0IHJlc3QgPSB7IC4uLmZpbHRlcmVkIH07XG4gIGRlbGV0ZSByZXN0Wydjb250ZXh0LnBhZ2VfaWQnXTtcbiAgcmV0dXJuIHJlc3Q7XG59O1xuXG4vLyBFeHRyYWN0IHdoaXRlbGlzdGVkIHN1Yi1maWVsZHMgZnJvbSB0cmlnZ2VyLndpZGdldF9pbnRlcmFjdGlvbl9kYXRhXG5jb25zdCBwcm9qZWN0V2lkZ2V0SW50ZXJhY3Rpb25EYXRhID0gKFxuICBldmVudFByb3BzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbiAgY29uZmlnOiBHQTRQcm9qZWN0ZWRFdmVudENvbmZpZyxcbik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgaWYgKCFjb25maWcud2lkZ2V0SW50ZXJhY3Rpb25EYXRhUHJvamVjdGlvbnMpIHtcbiAgICByZXR1cm4ge307XG4gIH1cblxuICBjb25zdCBpbnRlcmFjdGlvbiA9IGV2ZW50UHJvcHNbJ3RyaWdnZXIud2lkZ2V0X2ludGVyYWN0aW9uJ107XG4gIGlmICh0eXBlb2YgaW50ZXJhY3Rpb24gIT09ICdzdHJpbmcnKSB7XG4gICAgcmV0dXJuIHt9O1xuICB9XG5cbiAgY29uc3QgcHJvamVjdGlvbk1hcCA9IGNvbmZpZy53aWRnZXRJbnRlcmFjdGlvbkRhdGFQcm9qZWN0aW9uc1tpbnRlcmFjdGlvbl07XG4gIGlmICghcHJvamVjdGlvbk1hcCkge1xuICAgIHJldHVybiB7fTtcbiAgfVxuXG4gIGNvbnN0IGludGVyYWN0aW9uRGF0YSA9IGV2ZW50UHJvcHNbJ3RyaWdnZXIud2lkZ2V0X2ludGVyYWN0aW9uX2RhdGEnXTtcbiAgaWYgKFxuICAgIGludGVyYWN0aW9uRGF0YSA9PT0gbnVsbCB8fFxuICAgIGludGVyYWN0aW9uRGF0YSA9PT0gdW5kZWZpbmVkIHx8XG4gICAgdHlwZW9mIGludGVyYWN0aW9uRGF0YSAhPT0gJ29iamVjdCdcbiAgKSB7XG4gICAgcmV0dXJuIHt9O1xuICB9XG5cbiAgY29uc3QgZGF0YSA9IGludGVyYWN0aW9uRGF0YSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IFtnYUtleSwgc291cmNlRmllbGRdIG9mIE9iamVjdC5lbnRyaWVzKHByb2plY3Rpb25NYXApKSB7XG4gICAgaWYgKHNvdXJjZUZpZWxkIGluIGRhdGEpIHtcbiAgICAgIHJlc3VsdFtnYUtleV0gPSBkYXRhW3NvdXJjZUZpZWxkXTtcbiAgICB9XG4gIH1cbiAgcmV0dXJuIHJlc3VsdDtcbn07XG5cbmNvbnN0IHRydW5jYXRlU3RyaW5nID0gKHZhbHVlOiB1bmtub3duKTogdW5rbm93biA9PiB7XG4gIGlmICh0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnICYmIHZhbHVlLmxlbmd0aCA+IDEwMCkge1xuICAgIHJldHVybiBgJHt2YWx1ZS5zdWJzdHJpbmcoMCwgOTcpfS4uLmA7XG4gIH1cbiAgcmV0dXJuIHZhbHVlO1xufTtcblxuY29uc3QgdHJ1bmNhdGVWYWx1ZXMgPSAob2JqOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0+IHtcbiAgY29uc3QgcmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhvYmopKSB7XG4gICAgcmVzdWx0W2tleV0gPSB0cnVuY2F0ZVN0cmluZyh2YWx1ZSk7XG4gIH1cbiAgcmV0dXJuIHJlc3VsdDtcbn07XG5cbmNvbnN0IHB1c2hUb0RhdGFMYXllciA9IChnYUV2ZW50OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPik6IHZvaWQgPT4ge1xuICBpZiAodHlwZW9mIHdpbmRvdyAhPT0gJ3VuZGVmaW5lZCcgJiYgd2luZG93LmRhdGFMYXllcikge1xuICAgIHdpbmRvdy5kYXRhTGF5ZXIucHVzaChnYUV2ZW50KTtcbiAgfVxufTtcblxuZXhwb3J0IGNvbnN0IHByb2plY3RUb0dBNCA9IChcbiAgZXZlbnROYW1lOiBFbnZpdmVNZXRyaWNzRXZlbnROYW1lLFxuICBldmVudFByb3BzPzogUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4pOiB2b2lkID0+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCBzY2hlbWFFbnRyeSA9IEdBNF9FVkVOVF9TQ0hFTUFbZXZlbnROYW1lXTtcblxuICAgIGlmIChzY2hlbWFFbnRyeS5nYUV2ZW50TmFtZSA9PT0gbnVsbCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IGNvbmZpZyA9IHNjaGVtYUVudHJ5O1xuICAgIGNvbnN0IHByb3BzID0gZXZlbnRQcm9wcyA/PyB7fTtcblxuICAgIGxldCBmaWx0ZXJlZCA9IGZpbHRlclRvU2NoZW1hKHByb3BzLCBjb25maWcuYWxsb3dlZEZpZWxkcyk7XG4gICAgZmlsdGVyZWQgPSBzYW5pdGl6ZVBhZ2VJZChmaWx0ZXJlZCk7XG5cbiAgICBjb25zdCBpbnRlcmFjdGlvbkZpZWxkcyA9IHByb2plY3RXaWRnZXRJbnRlcmFjdGlvbkRhdGEocHJvcHMsIGNvbmZpZyk7XG5cbiAgICBjb25zdCBmbGF0UGFyYW1zID0ge1xuICAgICAgLi4uZmxhdHRlbkRvdEtleXMoZmlsdGVyZWQpLFxuICAgICAgLi4uaW50ZXJhY3Rpb25GaWVsZHMsXG4gICAgfTtcblxuICAgIGNvbnN0IHRydW5jYXRlZFBhcmFtcyA9IHRydW5jYXRlVmFsdWVzKGZsYXRQYXJhbXMpO1xuXG4gICAgcHVzaFRvRGF0YUxheWVyKHtcbiAgICAgIGV2ZW50OiBjb25maWcuZ2FFdmVudE5hbWUsXG4gICAgICAuLi50cnVuY2F0ZWRQYXJhbXMsXG4gICAgfSk7XG4gIH0gY2F0Y2ggKGVycikge1xuICAgIGxvZ2dlci5sb2dFcnJvcignRXJyb3IgcHJvamVjdGluZyBldmVudCB0byBHQTQnLCBlcnIsIHtcbiAgICAgIGV2ZW50TmFtZSxcbiAgICB9KTtcbiAgfVxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7OztBQUlBLE1BQU0sU0FBUyxJQUFJQSxlQUFPLHVCQUF1QjtBQUVqRCxNQUFNLGtCQUNKLFlBQ0Esa0JBQzRCO0NBQzVCLE1BQU1DLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLFNBQVMsY0FDbEIsS0FBSSxTQUFTLFdBQ1gsUUFBTyxTQUFTLFdBQVc7QUFHL0IsUUFBTzs7QUFJVCxNQUFNLGtCQUFrQixRQUEwRDtDQUNoRixNQUFNQSxTQUFrQyxFQUFFO0FBQzFDLE1BQUssTUFBTSxDQUFDLEtBQUssVUFBVSxPQUFPLFFBQVEsSUFBSSxDQUM1QyxRQUFPLElBQUksUUFBUSxPQUFPLElBQUksSUFBSTtBQUVwQyxRQUFPOztBQVNULE1BQU0sa0JBQWtCLGFBQStEO0NBQ3JGLE1BQU0sV0FBVyxTQUFTO0FBQzFCLEtBQUksYUFBYSxTQUFTLGFBQWEsTUFDckMsUUFBTztDQUVULE1BQU0sT0FBTyxFQUFFLEdBQUcsVUFBVTtBQUM1QixRQUFPLEtBQUs7QUFDWixRQUFPOztBQUlULE1BQU0sZ0NBQ0osWUFDQSxXQUM0QjtBQUM1QixLQUFJLENBQUMsT0FBTyxpQ0FDVixRQUFPLEVBQUU7Q0FHWCxNQUFNLGNBQWMsV0FBVztBQUMvQixLQUFJLE9BQU8sZ0JBQWdCLFNBQ3pCLFFBQU8sRUFBRTtDQUdYLE1BQU0sZ0JBQWdCLE9BQU8saUNBQWlDO0FBQzlELEtBQUksQ0FBQyxjQUNILFFBQU8sRUFBRTtDQUdYLE1BQU0sa0JBQWtCLFdBQVc7QUFDbkMsS0FDRSxvQkFBb0IsUUFDcEIsb0JBQW9CLFVBQ3BCLE9BQU8sb0JBQW9CLFNBRTNCLFFBQU8sRUFBRTtDQUdYLE1BQU0sT0FBTztDQUNiLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsT0FBTyxnQkFBZ0IsT0FBTyxRQUFRLGNBQWMsQ0FDOUQsS0FBSSxlQUFlLEtBQ2pCLFFBQU8sU0FBUyxLQUFLO0FBR3pCLFFBQU87O0FBR1QsTUFBTSxrQkFBa0IsVUFBNEI7QUFDbEQsS0FBSSxPQUFPLFVBQVUsWUFBWSxNQUFNLFNBQVMsSUFDOUMsUUFBTyxHQUFHLE1BQU0sVUFBVSxHQUFHLEdBQUcsQ0FBQztBQUVuQyxRQUFPOztBQUdULE1BQU0sa0JBQWtCLFFBQTBEO0NBQ2hGLE1BQU1BLFNBQWtDLEVBQUU7QUFDMUMsTUFBSyxNQUFNLENBQUMsS0FBSyxVQUFVLE9BQU8sUUFBUSxJQUFJLENBQzVDLFFBQU8sT0FBTyxlQUFlLE1BQU07QUFFckMsUUFBTzs7QUFHVCxNQUFNLG1CQUFtQixZQUEyQztBQUNsRSxLQUFJLE9BQU8sV0FBVyxlQUFlLE9BQU8sVUFDMUMsUUFBTyxVQUFVLEtBQUssUUFBUTs7QUFJbEMsTUFBYSxnQkFDWCxXQUNBLGVBQ1M7QUFDVCxLQUFJO0VBQ0YsTUFBTSxjQUFjLGlCQUFpQjtBQUVyQyxNQUFJLFlBQVksZ0JBQWdCLEtBQzlCO0VBR0YsTUFBTSxTQUFTO0VBQ2YsTUFBTSxRQUFRLGNBQWMsRUFBRTtFQUU5QixJQUFJLFdBQVcsZUFBZSxPQUFPLE9BQU8sY0FBYztBQUMxRCxhQUFXLGVBQWUsU0FBUztFQUVuQyxNQUFNLG9CQUFvQiw2QkFBNkIsT0FBTyxPQUFPO0VBT3JFLE1BQU0sa0JBQWtCLGVBTEw7R0FDakIsR0FBRyxlQUFlLFNBQVM7R0FDM0IsR0FBRztHQUNKLENBRWlEO0FBRWxELGtCQUFnQjtHQUNkLE9BQU8sT0FBTztHQUNkLEdBQUc7R0FDSixDQUFDO1VBQ0ssS0FBSztBQUNaLFNBQU8sU0FBUyxpQ0FBaUMsS0FBSyxFQUNwRCxXQUNELENBQUMifQ==
98
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ga4ProjectionService.js","names":["Logger","result: Record<string, unknown>","current: unknown","filtered: Record<string, unknown>"],"sources":["../../../src/services/ga4ProjectionService/ga4ProjectionService.ts"],"sourcesContent":["import Logger from 'src/application/logging/logger';\nimport { EnviveMetricsEventName } from '../amplitudeService/eventNames';\nimport { GA4ProjectedEventConfig, GA4_EVENT_SCHEMA } from './ga4EventSchema';\n\nconst logger = new Logger('ga4ProjectionService');\n\nconst filterToSchema = (\n  eventProps: Record<string, unknown>,\n  allowedFields: readonly string[],\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const field of allowedFields) {\n    if (field in eventProps) {\n      result[field] = eventProps[field];\n    }\n  }\n  return result;\n};\n\n// Flatten only true top-level nested objects (keys with no dots) one level deep.\n// Keys that already contain dots are left intact so that e.g.\n// `trigger.widget_interaction_data: { ... }` is preserved for downstream extraction.\nconst flattenOneLevel = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    if (\n      !key.includes('.') &&\n      value !== null &&\n      typeof value === 'object' &&\n      !Array.isArray(value)\n    ) {\n      for (const [subKey, subValue] of Object.entries(value as Record<string, unknown>)) {\n        result[`${key}.${subKey}`] = subValue;\n      }\n    } else {\n      result[key] = value;\n    }\n  }\n  return result;\n};\n\n// Apply a GA4 projection map { ga4Key: sourceKey } to extract and rename fields.\nconst filterWithProjections = (\n  eventProps: Record<string, unknown>,\n  projections: Record<string, string>,\n): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [ga4Key, sourceKey] of Object.entries(projections)) {\n    if (sourceKey in eventProps) {\n      result[ga4Key] = eventProps[sourceKey];\n    }\n  }\n  return result;\n};\n\n// \"context.page_type\" → \"context_page_type\"\nconst flattenDotKeys = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key.replace(/\\./g, '_')] = value;\n  }\n  return result;\n};\n\n// Omit context.page_id for non-pdp/plp page types. The current implementation for context.page_id is:\n// PDP: product_id\n// PLP: plp_id\n// Search: search query\n// Other: page url\n// We want to omit all but pdp and plp page types to provide a clear, consistent interface for merchants.\n// Handles both dot-notation keys (allowedFields path) and renamed GA4 keys (fieldProjections path).\nconst sanitizePageId = (filtered: Record<string, unknown>): Record<string, unknown> => {\n  const pageType = filtered['page_type'] ?? filtered['context.page_type'];\n  if (pageType === 'pdp' || pageType === 'plp') {\n    return filtered;\n  }\n  const rest = { ...filtered };\n  delete rest['page_id'];\n  delete rest['context.page_id'];\n  return rest;\n};\n\nconst getNestedValue = (obj: Record<string, unknown>, path: string): unknown => {\n  const parts = path.split('.');\n  let current: unknown = obj;\n  for (const part of parts) {\n    if (current === null || typeof current !== 'object') return undefined;\n    current = (current as Record<string, unknown>)[part];\n  }\n  return current;\n};\n\n// Extract whitelisted sub-fields from trigger.widget_interaction_data\nconst projectWidgetInteractionData = (\n  eventProps: Record<string, unknown>,\n  config: GA4ProjectedEventConfig,\n): Record<string, unknown> => {\n  if (!config.widgetInteractionDataProjections) {\n    return {};\n  }\n\n  const interaction = eventProps['trigger.widget_interaction'];\n  if (typeof interaction !== 'string') {\n    return {};\n  }\n\n  const projectionMap = config.widgetInteractionDataProjections[interaction];\n  if (!projectionMap) {\n    return {};\n  }\n\n  const interactionData = eventProps['trigger.widget_interaction_data'];\n  if (\n    interactionData === null ||\n    interactionData === undefined ||\n    typeof interactionData !== 'object'\n  ) {\n    return {};\n  }\n\n  const data = interactionData as Record<string, unknown>;\n  const result: Record<string, unknown> = {};\n  for (const [gaKey, sourcePath] of Object.entries(projectionMap)) {\n    const value = getNestedValue(data, sourcePath);\n    if (value !== undefined) {\n      result[gaKey] = value;\n    }\n  }\n  return result;\n};\n\nconst truncateString = (value: unknown): unknown => {\n  if (typeof value === 'string' && value.length > 100) {\n    return `${value.substring(0, 97)}...`;\n  }\n  return value;\n};\n\nconst truncateValues = (obj: Record<string, unknown>): Record<string, unknown> => {\n  const result: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(obj)) {\n    result[key] = truncateString(value);\n  }\n  return result;\n};\n\nconst pushToDataLayer = (gaEvent: Record<string, unknown>): void => {\n  if (typeof window !== 'undefined' && window.dataLayer) {\n    window.dataLayer.push(gaEvent);\n  }\n};\n\nexport const projectToGA4 = (\n  eventName: EnviveMetricsEventName,\n  eventProps?: Record<string, unknown>,\n): void => {\n  try {\n    const schemaEntry = GA4_EVENT_SCHEMA[eventName];\n\n    if (schemaEntry.gaEventName === null) {\n      return;\n    }\n\n    const config = schemaEntry;\n    const props = flattenOneLevel(eventProps ?? {});\n\n    let filtered: Record<string, unknown>;\n    if (config.fieldProjections) {\n      filtered = filterWithProjections(props, config.fieldProjections);\n    } else {\n      filtered = filterToSchema(props, config.allowedFields);\n    }\n    filtered = sanitizePageId(filtered);\n\n    const interactionFields = projectWidgetInteractionData(props, config);\n\n    const flatParams = {\n      ...flattenDotKeys(filtered),\n      ...interactionFields,\n    };\n\n    const truncatedParams = truncateValues(flatParams);\n\n    pushToDataLayer({\n      event: config.gaEventName,\n      ...truncatedParams,\n    });\n  } catch (err) {\n    logger.logError('Error projecting event to GA4', err, {\n      eventName,\n    });\n  }\n};\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@envive-ai/react-hooks",
3
- "version": "0.3.21",
3
+ "version": "0.3.23",
4
4
  "description": "React hooks for connecting to Envive AI services.",
5
5
  "keywords": [
6
6
  "react",
@@ -27,4 +27,5 @@ export enum FeatureGates {
27
27
  IsAiSuggestionsVariantEnabled = 'is_ai_suggestions_variant_enabled',
28
28
  IsAiSuggestionsVariantAEnabled = 'is_ai_suggestions_variant_a_enabled',
29
29
  IsAiSuggestionsVariantBEnabled = 'is_ai_suggestions_variant_b_enabled',
30
+ IsVoiceInputEnabled = 'is_voice_input_enabled',
30
31
  }
@@ -26,6 +26,7 @@ export const initializedAtom = atom<boolean>(false);
26
26
  export const chatIsOpenAtom = atom<boolean>(false);
27
27
  export const requestFailureAtom = atom<boolean>(false);
28
28
  export const formSubmitAtom = atom<FormSubmittedAttributes>();
29
+ export const listeningToSpeechAtom = atom<boolean>(false);
29
30
  export const chatOnToggleAtom = atom(
30
31
  null,
31
32
  (
@@ -20,6 +20,7 @@ import { AmplitudeProvider, SpiffyMetricsEventName, useAmplitude } from '../ampl
20
20
  const mockTrackEvent = vi.fn().mockResolvedValue(undefined);
21
21
  const mockSetSupplementalDefaultProps = vi.fn();
22
22
  const mockIsReady = vi.fn().mockReturnValue(true);
23
+ const mockIsMockApiKey = vi.fn().mockReturnValue(false);
23
24
 
24
25
  vi.mock('src/services/amplitudeService/amplitudeService', async () => {
25
26
  const actual = await vi.importActual<
@@ -34,6 +35,10 @@ vi.mock('src/services/amplitudeService/amplitudeService', async () => {
34
35
  get isReady(): boolean {
35
36
  return mockIsReady();
36
37
  }
38
+
39
+ get isMockApiKey(): boolean {
40
+ return mockIsMockApiKey();
41
+ }
37
42
  }
38
43
 
39
44
  function MockAmplitudeServiceConstructor() {
@@ -55,6 +60,7 @@ const MockAmplitudeComponent: React.FC = () => {
55
60
  return (
56
61
  <div data-testid="amplitude-component">
57
62
  <div data-testid="is-ready">{amplitude.isReady.toString()}</div>
63
+ <div data-testid="is-mock-mode">{amplitude.isMockMode.toString()}</div>
58
64
  <button
59
65
  data-testid="track-event-button"
60
66
  type="button"
@@ -213,6 +219,7 @@ describe('AmplitudeProvider - React Context Integration', () => {
213
219
  mockTrackEvent.mockClear();
214
220
  mockSetSupplementalDefaultProps.mockClear();
215
221
  mockIsReady.mockReturnValue(true);
222
+ mockIsMockApiKey.mockReturnValue(false);
216
223
  if (AmplitudeService && typeof AmplitudeService === 'function') {
217
224
  (AmplitudeService as unknown as ReturnType<typeof vi.fn>).mockClear();
218
225
  }
@@ -332,38 +339,35 @@ describe('AmplitudeProvider - React Context Integration', () => {
332
339
  });
333
340
  });
334
341
 
335
- it('should not render children when not ready', async () => {
336
- // Create a provider without required dependencies
337
- const IncompleteWrapper: React.FC<{ children: React.ReactNode }> = ({ children }) => {
338
- return (
339
- <Provider>
340
- <EnviveConfigProvider
341
- identifyingPrefix="test"
342
- orgShortName="test-org"
343
- // Missing amplitudeApiKey
344
- featureGates={[]}
345
- >
346
- <LocalStorageProvider>
347
- <UserIdentityProvider userIdService={testUserIdService}>
348
- <FeatureFlagServiceProvider featureFlagService={testFeatureFlagService}>
349
- <AmplitudeProvider>{children}</AmplitudeProvider>
350
- </FeatureFlagServiceProvider>
351
- </UserIdentityProvider>
352
- </LocalStorageProvider>
353
- </EnviveConfigProvider>
354
- </Provider>
355
- );
356
- };
342
+ it('should render children with isReady=false when isMockMode=true', async () => {
343
+ mockIsReady.mockReturnValue(false);
344
+ mockIsMockApiKey.mockReturnValue(true);
345
+
346
+ render(
347
+ <TestWrapper amplitudeApiKey="mock-amplitude-key">
348
+ <MockAmplitudeComponent />
349
+ </TestWrapper>,
350
+ );
351
+
352
+ await waitFor(() => {
353
+ expect(screen.getByTestId('amplitude-component')).toBeInTheDocument();
354
+ expect(screen.getByTestId('is-ready').textContent).toBe('false');
355
+ expect(screen.getByTestId('is-mock-mode').textContent).toBe('true');
356
+ });
357
+ });
358
+
359
+ it('should NOT render children when isReady=false and isMockMode=false', async () => {
360
+ mockIsReady.mockReturnValue(false);
361
+ mockIsMockApiKey.mockReturnValue(false);
357
362
 
358
363
  render(
359
- <IncompleteWrapper>
360
- <div data-testid="should-not-render">Should not render</div>
361
- </IncompleteWrapper>,
364
+ <TestWrapper>
365
+ <MockAmplitudeComponent />
366
+ </TestWrapper>,
362
367
  );
363
368
 
364
- // AmplitudeProvider returns null when not ready, so children should not render
365
369
  await waitFor(() => {
366
- expect(screen.queryByTestId('should-not-render')).not.toBeInTheDocument();
370
+ expect(screen.queryByTestId('amplitude-component')).not.toBeInTheDocument();
367
371
  });
368
372
  });
369
373
  });
@@ -23,6 +23,7 @@ export { EnviveMetricsEventName, SpiffyMetricsEventName };
23
23
  interface AmplitudeContextType {
24
24
  trackEvent: (params: TrackEventParams) => Promise<void>;
25
25
  isReady: boolean;
26
+ isMockMode: boolean;
26
27
  setSupplementalDefaultProps: (props: Record<string, unknown>) => void;
27
28
  }
28
29
 
@@ -55,6 +56,7 @@ export const AmplitudeProvider: React.FC<{
55
56
  const [service, setService] = useState<AmplitudeService | null>(null);
56
57
 
57
58
  const isReady = Boolean(userId && service && service.isReady);
59
+ const isMockMode = Boolean(service?.isMockApiKey);
58
60
 
59
61
  // Create service instance when dependencies are ready
60
62
  useEffect(() => {
@@ -109,16 +111,17 @@ export const AmplitudeProvider: React.FC<{
109
111
  }
110
112
  },
111
113
  isReady,
114
+ isMockMode,
112
115
  setSupplementalDefaultProps: (props: Record<string, unknown>) => {
113
116
  if (service) {
114
117
  service.setSupplementalDefaultProps(props);
115
118
  }
116
119
  },
117
120
  }),
118
- [service, isReady],
121
+ [service, isReady, isMockMode],
119
122
  );
120
123
 
121
- if (!isReady) {
124
+ if (!isReady && !isMockMode) {
122
125
  return null;
123
126
  }
124
127
 
@@ -219,10 +219,18 @@ export const HardcopyProvider: React.FC<HardcopyProviderProps> = ({
219
219
  const { featureFlagService } = useFeatureFlagService();
220
220
  const getHardcopyFromBackend = useCallback(
221
221
  async (request: HardcopyRequest): Promise<HardcopyResponse> => {
222
- if (hardcopyOverride?.[request.widgetType]) {
222
+ const fallbackWidgetType =
223
+ request.widgetType === WidgetTypeV3.ProductCardV3
224
+ ? WidgetTypeV3.ImagePromptCardV3
225
+ : request.widgetType;
226
+
227
+ const overrideEntry =
228
+ hardcopyOverride?.[request.widgetType] ?? hardcopyOverride?.[fallbackWidgetType];
229
+ if (overrideEntry) {
223
230
  logger.logDebug('using hardcopy override', request.widgetType);
224
- return hardcopyOverride[request.widgetType]!;
231
+ return overrideEntry;
225
232
  }
233
+
226
234
  const overrideConfigVersion =
227
235
  getQueryParam('spiffy_config_version') ||
228
236
  getQueryParam('envive_config_version') ||
@@ -19,6 +19,16 @@ vi.spyOn(Logger.prototype, 'logInfo').mockImplementation(() => {});
19
19
  vi.spyOn(Logger.prototype, 'logWarn').mockImplementation(() => {});
20
20
  vi.spyOn(Logger.prototype, 'logError').mockImplementation(() => {});
21
21
 
22
+ vi.mock('src/contexts/amplitudeContext', () => ({
23
+ useAmplitude: () => ({
24
+ trackEvent: vi.fn(),
25
+ isReady: true,
26
+ }),
27
+ EnviveMetricsEventName: {
28
+ PageViewed: 'Page Viewed',
29
+ },
30
+ }));
31
+
22
32
  // Mock CommerceApiClient
23
33
  const mockResolveUrl = vi.fn();
24
34
  vi.mock('src/application/commerce-api', () => ({
@@ -14,6 +14,7 @@ import {
14
14
  import { hasParsedVariantInfoAtom } from 'src/atoms/app';
15
15
  import { analyticsContextAtom } from 'src/atoms/app/variant';
16
16
  import { formSubmitAtom, replyEventCategoryAtom } from 'src/atoms/chat';
17
+ import { suggestionsAtom } from 'src/atoms/chat/chatState';
17
18
  import { queueUserEventAtom } from 'src/atoms/chat/messageQueue';
18
19
  import { useWidgetInteraction } from 'src/hooks/WidgetInteraction';
19
20
  import {
@@ -56,6 +57,7 @@ export const useSalesAgentChatAPI = (widget?: WidgetInteractionComponent) => {
56
57
  const context = useAtomValue(analyticsContextAtom);
57
58
  const hasParsedVariantInfo = useAtomValue(hasParsedVariantInfoAtom);
58
59
  const queueUserEvent = useSetAtom(queueUserEventAtom);
60
+ const setSuggestions = useSetAtom(suggestionsAtom);
59
61
  const setReplyEventCategory = useSetAtom(replyEventCategoryAtom);
60
62
  const setFormSubmit = useSetAtom(formSubmitAtom);
61
63
  const { trackEvent } = useAmplitude();
@@ -164,9 +166,10 @@ export const useSalesAgentChatAPI = (widget?: WidgetInteractionComponent) => {
164
166
  content: suggestion.content,
165
167
  },
166
168
  };
169
+ setSuggestions([]);
167
170
  queueUserEvent(event);
168
171
  },
169
- [queueUserEvent, trackEvent, context],
172
+ [queueUserEvent, setSuggestions, trackEvent, context],
170
173
  );
171
174
  const onTypedMessageSubmitted = useCallback(
172
175
  ({
@@ -229,9 +232,10 @@ export const useSalesAgentChatAPI = (widget?: WidgetInteractionComponent) => {
229
232
  userTyped,
230
233
  },
231
234
  };
235
+ setSuggestions([]);
232
236
  queueUserEvent(event);
233
237
  },
234
- [queueUserEvent, trackEvent, context],
238
+ [queueUserEvent, setSuggestions, trackEvent, context],
235
239
  );
236
240
  const onFormResponseSubmitted = useCallback(
237
241
  (form: FormSubmittedAttributes) => {
@@ -156,6 +156,7 @@ type LookAndFeelConfig = {
156
156
  chatHeaderLogoLightSrc: string;
157
157
  widgetLogoSrc: string;
158
158
  hideWidgetLogo: boolean;
159
+ voiceInputEnabled?: boolean;
159
160
  elementRadius: number;
160
161
  imageAspectRatio: ImageAspectRatio;
161
162
  buttonColors: ButtonColors;
@@ -29,10 +29,6 @@ import { UserIdentityService } from 'src/services/userIdentityService';
29
29
  import { useSearch } from '../useSearch';
30
30
 
31
31
  // Mock dependencies
32
- vi.mock('src/hooks/TrackComponentVisibleEvent/useTrackComponentVisibleEvent', () => ({
33
- useTrackComponentVisibleEvent: vi.fn(),
34
- }));
35
-
36
32
  vi.mock('src/hooks/Intersection/useIntersection', () => ({
37
33
  useIntersection: vi.fn(() => false),
38
34
  }));