1
1
'use client' ;
2
- import type { RouteExtended } from '@lifi/sdk' ;
3
- import { type Route } from '@lifi/sdk' ;
4
-
5
2
import { MultisigConfirmationModal } from '@/components/MultisigConfirmationModal' ;
6
3
import { MultisigConnectedAlert } from '@/components/MultisigConnectedAlert' ;
7
4
import {
@@ -16,6 +13,8 @@ import { useActiveTabStore } from '@/stores/activeTab';
16
13
import { useChainTokenSelectionStore } from '@/stores/chainTokenSelection' ;
17
14
import { useMenuStore } from '@/stores/menu' ;
18
15
import { useMultisigStore } from '@/stores/multisig' ;
16
+ import type { RouteExtended } from '@lifi/sdk' ;
17
+ import { type Route } from '@lifi/sdk' ;
19
18
import type {
20
19
ChainTokenSelected ,
21
20
ContactSupport ,
@@ -24,13 +23,21 @@ import type {
24
23
} from '@lifi/widget' ;
25
24
import { WidgetEvent , useWidgetEvents } from '@lifi/widget' ;
26
25
import { useEffect , useRef , useState } from 'react' ;
26
+ import { shallowEqualObjects } from 'shallow-equal' ;
27
+ import type { JumperEventData } from 'src/hooks/useJumperTracking' ;
28
+ import type { TransformedRoute } from 'src/types/internal' ;
29
+ import { calcPriceImpact } from 'src/utils/calcPriceImpact' ;
27
30
import { handleTransactionDetails } from 'src/utils/routesInterpreterUtils' ;
28
31
29
32
export function WidgetEvents ( ) {
30
- const lastTxHashRef = useRef < string > ( ) ;
33
+ const previousRoutesRef = useRef < JumperEventData > ( { } ) ;
31
34
const { activeTab } = useActiveTabStore ( ) ;
32
- const { setDestinationChainToken, setSourceChainToken } =
33
- useChainTokenSelectionStore ( ) ;
35
+ const {
36
+ sourceChainToken,
37
+ destinationChainToken,
38
+ setDestinationChainToken,
39
+ setSourceChainToken,
40
+ } = useChainTokenSelectionStore ( ) ;
34
41
const { trackTransaction, trackEvent } = useUserTracking ( ) ;
35
42
const [ setSupportModalState ] = useMenuStore ( ( state ) => [
36
43
state . setSupportModalState ,
@@ -68,36 +75,22 @@ export function WidgetEvents() {
68
75
69
76
const onRouteExecutionUpdated = async ( update : RouteExecutionUpdate ) => {
70
77
// check if multisig and open the modal
71
- const data = handleTransactionDetails ( update . route , {
72
- [ TrackingEventParameter . Action ] : 'execution_updated' ,
73
- } ) ;
74
78
const isMultisigRouteActive = shouldOpenMultisigSignatureModal (
75
79
update . route ,
76
80
) ;
77
-
78
81
if ( isMultisigRouteActive ) {
79
82
setIsMultiSigConfirmationModalOpen ( true ) ;
80
83
}
81
-
82
- if ( update . process && update . route ) {
83
- if ( update . process . txHash !== lastTxHashRef . current ) {
84
- lastTxHashRef . current = update . process . txHash ;
85
- // trackTransaction({
86
- // category: TrackingCategory.WidgetEvent,
87
- // action: TrackingAction.OnRouteExecutionUpdated,
88
- // label: 'execution_update',
89
- // data,
90
- // enableAddressable: true,
91
- // });
92
- }
93
- }
94
84
} ;
85
+
95
86
const onRouteExecutionCompleted = async ( route : Route ) => {
96
87
if ( route . id ) {
97
88
const data = handleTransactionDetails ( route , {
98
89
[ TrackingEventParameter . Action ] : 'execution_completed' ,
99
90
[ TrackingEventParameter . TransactionStatus ] : 'COMPLETED' ,
100
91
} ) ;
92
+ // reset ref obj
93
+ previousRoutesRef . current = { } ;
101
94
trackTransaction ( {
102
95
category : TrackingCategory . WidgetEvent ,
103
96
action : TrackingAction . OnRouteExecutionCompleted ,
@@ -108,6 +101,7 @@ export function WidgetEvents() {
108
101
} ) ;
109
102
}
110
103
} ;
104
+
111
105
const onRouteExecutionFailed = async ( update : RouteExecutionUpdate ) => {
112
106
const data = handleTransactionDetails ( update . route , {
113
107
[ TrackingEventParameter . Action ] : 'execution_failed' ,
@@ -118,6 +112,8 @@ export function WidgetEvents() {
118
112
[ TrackingEventParameter . IsFinal ] : true ,
119
113
[ TrackingEventParameter . ErrorCode ] : update . process . error ?. code || '' ,
120
114
} ) ;
115
+ // reset ref obj
116
+ previousRoutesRef . current = { } ;
121
117
trackTransaction ( {
122
118
category : TrackingCategory . WidgetEvent ,
123
119
action : TrackingAction . OnRouteExecutionFailed ,
@@ -204,17 +200,73 @@ export function WidgetEvents() {
204
200
} ;
205
201
206
202
const onAvailableRoutes = async ( availableRoutes : Route [ ] ) => {
207
- trackEvent ( {
208
- category : TrackingCategory . WidgetEvent ,
209
- action : TrackingAction . OnAvailableRoutes ,
210
- label : `routes_available` ,
211
- enableAddressable : true ,
212
- data : {
213
- [ TrackingEventParameter . AvailableRoutesCount ] : availableRoutes . length ,
214
- } ,
215
- } ) ;
216
- } ;
203
+ // current available routes
204
+ const newObj : JumperEventData = {
205
+ [ TrackingEventParameter . FromToken ] : sourceChainToken . tokenAddress || '' ,
206
+ [ TrackingEventParameter . FromChainId ] : sourceChainToken . chainId || '' ,
207
+ [ TrackingEventParameter . ToToken ] :
208
+ destinationChainToken . tokenAddress || '' ,
209
+ [ TrackingEventParameter . ToChainId ] : destinationChainToken . chainId || '' ,
210
+ } ;
217
211
212
+ // compare current availableRoutes with the previous one
213
+ const isSameObject = shallowEqualObjects (
214
+ previousRoutesRef . current ,
215
+ newObj ,
216
+ ) ;
217
+ // if the object has changed, then track the event
218
+ if (
219
+ ! isSameObject &&
220
+ previousRoutesRef . current &&
221
+ sourceChainToken . chainId &&
222
+ sourceChainToken . tokenAddress &&
223
+ destinationChainToken . chainId &&
224
+ destinationChainToken . tokenAddress
225
+ ) {
226
+ previousRoutesRef . current = newObj ;
227
+ const transformedRoutes = availableRoutes . reduce <
228
+ Record < number , TransformedRoute >
229
+ > ( ( acc , route , index ) => {
230
+ const priceImpact = calcPriceImpact ( route ) ;
231
+ acc [ index ] = {
232
+ [ TrackingEventParameter . NbOfSteps ] : route . steps . length ,
233
+ [ TrackingEventParameter . Steps ] : route . steps . map (
234
+ ( step ) => step . tool ,
235
+ ) ,
236
+ [ TrackingEventParameter . ToAmountUSD ] : Number ( route . toAmountUSD ) ,
237
+ [ TrackingEventParameter . GasCostUSD ] : route . steps . reduce (
238
+ ( acc , step ) =>
239
+ acc +
240
+ ( step . estimate . gasCosts ?. reduce (
241
+ ( sum , gasCost ) => sum + parseFloat ( gasCost . amountUSD ) ,
242
+ 0 ,
243
+ ) || 0 ) ,
244
+ 0 ,
245
+ ) ,
246
+ [ TrackingEventParameter . Time ] : route . steps . reduce (
247
+ ( acc , step ) => acc + step . estimate . executionDuration ,
248
+ 0 ,
249
+ ) ,
250
+ [ TrackingEventParameter . Slippage ] : priceImpact ,
251
+ } ;
252
+ return acc ;
253
+ } , { } ) ;
254
+ trackEvent ( {
255
+ category : TrackingCategory . WidgetEvent ,
256
+ action : TrackingAction . OnAvailableRoutes ,
257
+ label : `routes_available` ,
258
+ enableAddressable : true ,
259
+ data : {
260
+ ...newObj ,
261
+ [ TrackingEventParameter . FromAmountUSD ] : Number (
262
+ availableRoutes [ 0 ] . fromAmountUSD ,
263
+ ) ,
264
+ [ TrackingEventParameter . NbOfSteps ] : availableRoutes . length ,
265
+ [ TrackingEventParameter . Routes ] : transformedRoutes ,
266
+ } ,
267
+ } ) ;
268
+ }
269
+ } ;
218
270
widgetEvents . on ( WidgetEvent . RouteExecutionStarted , onRouteExecutionStarted ) ;
219
271
widgetEvents . on ( WidgetEvent . RouteExecutionUpdated , onRouteExecutionUpdated ) ;
220
272
widgetEvents . on (
@@ -237,6 +289,7 @@ export function WidgetEvents() {
237
289
WidgetEvent . DestinationChainTokenSelected ,
238
290
onDestinationChainTokenSelection ,
239
291
) ;
292
+
240
293
// widgetEvents.on(WidgetEvent.WidgetExpanded, onWidgetExpanded);
241
294
242
295
return ( ) => {
@@ -275,11 +328,14 @@ export function WidgetEvents() {
275
328
} ;
276
329
} , [
277
330
activeTab ,
331
+ destinationChainToken . chainId ,
332
+ destinationChainToken . tokenAddress ,
278
333
setDestinationChain ,
279
334
setDestinationChainToken ,
280
335
setSourceChainToken ,
281
336
setSupportModalState ,
282
337
shouldOpenMultisigSignatureModal ,
338
+ sourceChainToken ,
283
339
trackEvent ,
284
340
trackTransaction ,
285
341
widgetEvents ,
0 commit comments