|
1 |
| -import { getChecksumAddress, Provider } from "starknet"; |
2 |
| -import useSWR from "swr"; |
3 |
| -import { ERC20, ERC20Metadata } from "../erc20"; |
4 | 1 | import { CreditQuery, useCreditQuery } from "../api/cartridge";
|
5 |
| -import { erc20Metadata } from "@cartridge/presets"; |
6 |
| - |
7 |
| -export function useERC20Balance({ |
8 |
| - address, |
9 |
| - contractAddress, |
10 |
| - provider, |
11 |
| - interval, |
12 |
| - decimals = 5, |
13 |
| -}: { |
14 |
| - address?: string; |
15 |
| - contractAddress: string | string[]; |
16 |
| - provider?: Provider; |
17 |
| - interval: number | undefined; |
18 |
| - decimals?: number; |
19 |
| -}) { |
20 |
| - const { data: chainId } = useSWR(address && provider ? "chainId" : null, () => |
21 |
| - provider?.getChainId(), |
22 |
| - ); |
23 |
| - const { data: meta } = useSWR( |
24 |
| - chainId && erc20Metadata.length |
25 |
| - ? `erc20:metadata:${chainId}:${address}:${contractAddress}` |
26 |
| - : null, |
27 |
| - async () => { |
28 |
| - if (!provider || !address) return []; |
29 |
| - const contractList = Array.isArray(contractAddress) |
30 |
| - ? contractAddress |
31 |
| - : [contractAddress]; |
32 |
| - const erc20List = await Promise.allSettled( |
33 |
| - contractList.map((address) => |
34 |
| - new ERC20({ |
35 |
| - address, |
36 |
| - provider, |
37 |
| - logoUrl: erc20Metadata.find( |
38 |
| - (m) => |
39 |
| - getChecksumAddress(m.l2_token_address) === |
40 |
| - getChecksumAddress(address), |
41 |
| - )?.logo_url, |
42 |
| - }).init(), |
43 |
| - ), |
44 |
| - ); |
45 |
| - |
46 |
| - return erc20List |
47 |
| - .filter((res) => res.status === "fulfilled") |
48 |
| - .map((erc20) => erc20.value.metadata()); |
49 |
| - }, |
50 |
| - { fallbackData: [] }, |
51 |
| - ); |
52 |
| - |
53 |
| - const { data, isValidating, isLoading, error } = useSWR( |
54 |
| - meta.length && address |
55 |
| - ? `erc20:balance:${chainId}:${address}:${contractAddress}` |
56 |
| - : null, |
57 |
| - async () => { |
58 |
| - if (!address) return []; |
59 |
| - const values = await Promise.allSettled( |
60 |
| - meta.map((m) => m.instance.balanceOf(address)), |
61 |
| - ); |
62 |
| - |
63 |
| - return meta.reduce<{ balance: Balance; meta: ERC20Metadata }[]>( |
64 |
| - (prev, meta, i) => { |
65 |
| - const res = values[i]; |
66 |
| - if (res.status === "rejected") return prev; |
67 |
| - |
68 |
| - const value = res.value; |
69 |
| - const factor = 10 ** meta.decimals; |
70 |
| - const adjusted = parseFloat(value.toString()) / factor; |
71 |
| - // Round and remove insignificant trailing zeros |
72 |
| - const rounded = parseFloat(adjusted.toFixed(decimals)); |
73 |
| - const formatted = |
74 |
| - adjusted === rounded ? adjusted.toString() : `~${rounded}`; |
75 |
| - |
76 |
| - return [ |
77 |
| - ...prev, |
78 |
| - { |
79 |
| - balance: { |
80 |
| - value, |
81 |
| - formatted, |
82 |
| - }, |
83 |
| - meta, |
84 |
| - }, |
85 |
| - ]; |
86 |
| - }, |
87 |
| - [], |
88 |
| - ); |
89 |
| - }, |
90 |
| - { refreshInterval: interval, fallbackData: [] }, |
91 |
| - ); |
92 |
| - |
93 |
| - return { |
94 |
| - data, |
95 |
| - isFetching: isValidating, |
96 |
| - isLoading, |
97 |
| - error, |
98 |
| - }; |
99 |
| -} |
100 | 2 |
|
101 | 3 | export type Balance = {
|
102 | 4 | value: bigint;
|
|
0 commit comments