Skip to content

Commit 5ec37df

Browse files
authored
chore: update availableRoutes widget-event (#1313)
1 parent bae128e commit 5ec37df

12 files changed

+798
-917
lines changed

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@
4848
"react-error-boundary": "^4.0.13",
4949
"react-i18next": "^15.0.2",
5050
"react-router-dom": "^6.26.2",
51+
"shallow-equal": "^3.1.0",
5152
"sharp": "^0.33.5",
5253
"siwe": "^2.3.2",
5354
"typescript": "^5.6.2",

src/components/ProfilePage/ProfilePage.tsx

+2-4
Original file line numberDiff line numberDiff line change
@@ -13,10 +13,8 @@ import { QuestCarousel } from './QuestCarousel/QuestCarousel';
1313
import { QuestCompletedList } from './QuestsCompleted/QuestsCompletedList';
1414
import { Leaderboard } from './Leaderboard/Leaderboard';
1515
import { RewardsCarousel } from './Rewards/RewardsCarousel';
16-
import {
17-
AvailableRewards,
18-
useMerklRewardsOnCampaigns,
19-
} from 'src/hooks/useMerklRewardsOnCampaigns';
16+
import type { AvailableRewards } from 'src/hooks/useMerklRewardsOnCampaigns';
17+
import { useMerklRewardsOnCampaigns } from 'src/hooks/useMerklRewardsOnCampaigns';
2018
import { useMemo } from 'react';
2119

2220
const shouldHideComponent = (

src/components/Widgets/WidgetEvents.tsx

+89-33
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,4 @@
11
'use client';
2-
import type { RouteExtended } from '@lifi/sdk';
3-
import { type Route } from '@lifi/sdk';
4-
52
import { MultisigConfirmationModal } from '@/components/MultisigConfirmationModal';
63
import { MultisigConnectedAlert } from '@/components/MultisigConnectedAlert';
74
import {
@@ -16,6 +13,8 @@ import { useActiveTabStore } from '@/stores/activeTab';
1613
import { useChainTokenSelectionStore } from '@/stores/chainTokenSelection';
1714
import { useMenuStore } from '@/stores/menu';
1815
import { useMultisigStore } from '@/stores/multisig';
16+
import type { RouteExtended } from '@lifi/sdk';
17+
import { type Route } from '@lifi/sdk';
1918
import type {
2019
ChainTokenSelected,
2120
ContactSupport,
@@ -24,13 +23,21 @@ import type {
2423
} from '@lifi/widget';
2524
import { WidgetEvent, useWidgetEvents } from '@lifi/widget';
2625
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';
2730
import { handleTransactionDetails } from 'src/utils/routesInterpreterUtils';
2831

2932
export function WidgetEvents() {
30-
const lastTxHashRef = useRef<string>();
33+
const previousRoutesRef = useRef<JumperEventData>({});
3134
const { activeTab } = useActiveTabStore();
32-
const { setDestinationChainToken, setSourceChainToken } =
33-
useChainTokenSelectionStore();
35+
const {
36+
sourceChainToken,
37+
destinationChainToken,
38+
setDestinationChainToken,
39+
setSourceChainToken,
40+
} = useChainTokenSelectionStore();
3441
const { trackTransaction, trackEvent } = useUserTracking();
3542
const [setSupportModalState] = useMenuStore((state) => [
3643
state.setSupportModalState,
@@ -68,36 +75,22 @@ export function WidgetEvents() {
6875

6976
const onRouteExecutionUpdated = async (update: RouteExecutionUpdate) => {
7077
// check if multisig and open the modal
71-
const data = handleTransactionDetails(update.route, {
72-
[TrackingEventParameter.Action]: 'execution_updated',
73-
});
7478
const isMultisigRouteActive = shouldOpenMultisigSignatureModal(
7579
update.route,
7680
);
77-
7881
if (isMultisigRouteActive) {
7982
setIsMultiSigConfirmationModalOpen(true);
8083
}
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-
}
9484
};
85+
9586
const onRouteExecutionCompleted = async (route: Route) => {
9687
if (route.id) {
9788
const data = handleTransactionDetails(route, {
9889
[TrackingEventParameter.Action]: 'execution_completed',
9990
[TrackingEventParameter.TransactionStatus]: 'COMPLETED',
10091
});
92+
// reset ref obj
93+
previousRoutesRef.current = {};
10194
trackTransaction({
10295
category: TrackingCategory.WidgetEvent,
10396
action: TrackingAction.OnRouteExecutionCompleted,
@@ -108,6 +101,7 @@ export function WidgetEvents() {
108101
});
109102
}
110103
};
104+
111105
const onRouteExecutionFailed = async (update: RouteExecutionUpdate) => {
112106
const data = handleTransactionDetails(update.route, {
113107
[TrackingEventParameter.Action]: 'execution_failed',
@@ -118,6 +112,8 @@ export function WidgetEvents() {
118112
[TrackingEventParameter.IsFinal]: true,
119113
[TrackingEventParameter.ErrorCode]: update.process.error?.code || '',
120114
});
115+
// reset ref obj
116+
previousRoutesRef.current = {};
121117
trackTransaction({
122118
category: TrackingCategory.WidgetEvent,
123119
action: TrackingAction.OnRouteExecutionFailed,
@@ -204,17 +200,73 @@ export function WidgetEvents() {
204200
};
205201

206202
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+
};
217211

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+
};
218270
widgetEvents.on(WidgetEvent.RouteExecutionStarted, onRouteExecutionStarted);
219271
widgetEvents.on(WidgetEvent.RouteExecutionUpdated, onRouteExecutionUpdated);
220272
widgetEvents.on(
@@ -237,6 +289,7 @@ export function WidgetEvents() {
237289
WidgetEvent.DestinationChainTokenSelected,
238290
onDestinationChainTokenSelection,
239291
);
292+
240293
// widgetEvents.on(WidgetEvent.WidgetExpanded, onWidgetExpanded);
241294

242295
return () => {
@@ -275,11 +328,14 @@ export function WidgetEvents() {
275328
};
276329
}, [
277330
activeTab,
331+
destinationChainToken.chainId,
332+
destinationChainToken.tokenAddress,
278333
setDestinationChain,
279334
setDestinationChainToken,
280335
setSourceChainToken,
281336
setSupportModalState,
282337
shouldOpenMultisigSignatureModal,
338+
sourceChainToken,
283339
trackEvent,
284340
trackTransaction,
285341
widgetEvents,

src/const/trackingKeys.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -168,8 +168,12 @@ export enum TrackingEventParameter {
168168
ErrorCode = 'param_error_code',
169169
ErrorMessage = 'param_error_message',
170170
ValueLoss = 'param_value_loss',
171-
AvailableRoutesCount = 'param_available_routes_count',
172171
TransactionStatus = 'param_transaction_status',
172+
Routes = 'param_routes',
173+
NbOfSteps = 'param_nb_of_steps',
174+
Steps = 'param_steps',
175+
Time = 'param_time',
176+
Slippage = 'param_slippage',
173177

174178
// Blog
175179
ArticleCardId = 'param_article_card_id',

src/hooks/useJumperTracking.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -7,13 +7,19 @@ import {
77
JUMPER_ANALYTICS_EVENT,
88
JUMPER_ANALYTICS_TRANSACTION,
99
} from 'src/const/abi/jumperApiUrls';
10+
import type { TransformedRoute } from 'src/types/internal';
11+
12+
export type JumperEventData = {
13+
[key: string]: string | number | boolean | Record<number, TransformedRoute>;
14+
};
15+
1016
interface JumperDataTrackEventProps {
1117
category: string;
1218
action: string;
1319
label: string;
1420
url: string;
1521
value: number;
16-
data?: { [key: string]: string | number | boolean };
22+
data?: JumperEventData;
1723
isConnected: boolean;
1824
walletAddress?: string;
1925
browserFingerprint: string;

src/hooks/userTracking/useUserTracking.ts

+10-4
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import {
55
TrackingEventParameter,
66
} from '@/const/trackingKeys';
77
import { useAccounts } from '@/hooks/useAccounts';
8+
import type { JumperEventData } from '@/hooks/useJumperTracking';
89
import { useJumperTracking } from '@/hooks/useJumperTracking';
910
import { useSession } from '@/hooks/useSession';
1011
import type {
@@ -16,6 +17,7 @@ import { EventTrackingTool } from '@/types/userTracking';
1617
import type { Theme } from '@mui/material';
1718
import { useMediaQuery } from '@mui/material';
1819
import { useCallback, useEffect } from 'react';
20+
import type { TransformedRoute } from 'src/types/internal';
1921
import { useFingerprint } from '../useFingerprint';
2022

2123
const googleEvent = ({
@@ -27,7 +29,13 @@ const googleEvent = ({
2729
category: string;
2830
data?:
2931
| TrackTransactionDataProps
30-
| { [key: string]: string | number | boolean };
32+
| {
33+
[key: string]:
34+
| string
35+
| number
36+
| boolean
37+
| Record<number, TransformedRoute>;
38+
};
3139
}) => {
3240
typeof window !== 'undefined' &&
3341
window?.gtag('event', action, {
@@ -44,9 +52,7 @@ const addressableEvent = ({
4452
}: {
4553
action: string;
4654
label: string;
47-
data:
48-
| TrackTransactionDataProps
49-
| { [key: string]: string | number | boolean };
55+
data: TrackTransactionDataProps | JumperEventData;
5056
isConversion?: boolean;
5157
}) => {
5258
const dataArray = [];

src/types/internal.ts

+9
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import type { WidgetConfig, WidgetSubvariant } from '@lifi/widget';
33
import type { SxProps, Theme } from '@mui/material';
44
import type { MenuItemLinkType } from 'src/components/Menu';
55
import type { MenuKeysEnum } from 'src/const/menuKeys';
6+
import type { TrackingEventParameter } from 'src/const/trackingKeys';
67

78
declare global {
89
interface Window {
@@ -63,3 +64,11 @@ export interface DataItem {
6364
logoURI?: string;
6465
name: string;
6566
}
67+
export interface TransformedRoute {
68+
[TrackingEventParameter.NbOfSteps]: number;
69+
[TrackingEventParameter.Steps]: object;
70+
[TrackingEventParameter.ToAmountUSD]: number;
71+
[TrackingEventParameter.GasCostUSD]: number | null;
72+
[TrackingEventParameter.Time]: number;
73+
[TrackingEventParameter.Slippage]: number;
74+
}

src/types/userTracking.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import type { TrackingEventParameter } from 'src/const/trackingKeys';
2+
import type { JumperEventData } from 'src/hooks/useJumperTracking';
23

34
export enum EventTrackingTool {
45
GA,
@@ -14,7 +15,7 @@ export interface TrackEventProps {
1415
category: string;
1516
label: string;
1617
value?: number;
17-
data?: { [key: string]: string | number | boolean };
18+
data?: JumperEventData;
1819
disableTrackingTool?: EventTrackingTool[];
1920
enableAddressable?: boolean;
2021
isConversion?: boolean;

src/utils/calcPriceImpact.ts

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
import type { Route } from '@lifi/sdk';
2+
import { formatTokenAmount, formatTokenPrice } from './format';
3+
4+
export const calcPriceImpact = (route: Route) => {
5+
const fromTokenAmount = formatTokenAmount(
6+
BigInt(route.fromAmount),
7+
route.fromToken.decimals,
8+
);
9+
const fromTokenPrice = formatTokenPrice(
10+
fromTokenAmount,
11+
route.fromToken.priceUSD,
12+
);
13+
const toTokenAmount = formatTokenAmount(
14+
BigInt(route.toAmount),
15+
route.toToken.decimals,
16+
);
17+
const toTokenPrice =
18+
formatTokenPrice(toTokenAmount, route.toToken.priceUSD) || 0.01;
19+
20+
const priceImpact = (toTokenPrice / fromTokenPrice - 1).toFixed(6);
21+
22+
return Number(priceImpact);
23+
};

src/utils/format.ts

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { formatUnits } from './formatUnit';
2+
3+
export const precisionFormatter = new Intl.NumberFormat('en', {
4+
notation: 'standard',
5+
roundingPriority: 'morePrecision',
6+
maximumSignificantDigits: 4,
7+
maximumFractionDigits: 4,
8+
useGrouping: false,
9+
});
10+
11+
export function formatTokenAmount(amount: bigint = 0n, decimals: number) {
12+
const formattedAmount = amount ? formatUnits(amount, decimals) : '0';
13+
const parsedAmount = parseFloat(formattedAmount);
14+
if (parsedAmount === 0 || isNaN(Number(formattedAmount))) {
15+
return '0';
16+
}
17+
18+
return precisionFormatter.format(parsedAmount);
19+
}
20+
21+
export function formatTokenPrice(amount?: string, price?: string) {
22+
if (!amount || !price) {
23+
return 0;
24+
}
25+
if (isNaN(Number(amount)) || isNaN(Number(price))) {
26+
return 0;
27+
}
28+
return parseFloat(amount) * parseFloat(price);
29+
}

0 commit comments

Comments
 (0)