@metamask/assets-controllers 25.0.0 → 27.0.0
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.
- package/CHANGELOG.md +210 -39
- package/dist/AccountTrackerController.js +9 -236
- package/dist/AccountTrackerController.js.map +1 -1
- package/dist/AccountTrackerController.mjs +10 -0
- package/dist/AccountTrackerController.mjs.map +1 -0
- package/dist/AssetsContractController.js +17 -385
- package/dist/AssetsContractController.js.map +1 -1
- package/dist/AssetsContractController.mjs +18 -0
- package/dist/AssetsContractController.mjs.map +1 -0
- package/dist/CurrencyRateController.js +10 -157
- package/dist/CurrencyRateController.js.map +1 -1
- package/dist/CurrencyRateController.mjs +11 -0
- package/dist/CurrencyRateController.mjs.map +1 -0
- package/dist/NftController.js +15 -1039
- package/dist/NftController.js.map +1 -1
- package/dist/NftController.mjs +16 -0
- package/dist/NftController.mjs.map +1 -0
- package/dist/NftDetectionController.js +12 -215
- package/dist/NftDetectionController.js.map +1 -1
- package/dist/NftDetectionController.mjs +13 -0
- package/dist/NftDetectionController.mjs.map +1 -0
- package/dist/Standards/ERC20Standard.js +8 -144
- package/dist/Standards/ERC20Standard.js.map +1 -1
- package/dist/Standards/ERC20Standard.mjs +9 -0
- package/dist/Standards/ERC20Standard.mjs.map +1 -0
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.js +8 -217
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.js.map +1 -1
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.mjs +9 -0
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.mjs.map +1 -0
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.js +8 -175
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.js.map +1 -1
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.mjs +9 -0
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.mjs.map +1 -0
- package/dist/TokenBalancesController.js +11 -144
- package/dist/TokenBalancesController.js.map +1 -1
- package/dist/TokenBalancesController.mjs +12 -0
- package/dist/TokenBalancesController.mjs.map +1 -0
- package/dist/TokenDetectionController.js +16 -245
- package/dist/TokenDetectionController.js.map +1 -1
- package/dist/TokenDetectionController.mjs +17 -0
- package/dist/TokenDetectionController.mjs.map +1 -0
- package/dist/TokenListController.js +13 -250
- package/dist/TokenListController.js.map +1 -1
- package/dist/TokenListController.mjs +14 -0
- package/dist/TokenListController.mjs.map +1 -0
- package/dist/TokenRatesController.js +11 -328
- package/dist/TokenRatesController.js.map +1 -1
- package/dist/TokenRatesController.mjs +12 -0
- package/dist/TokenRatesController.mjs.map +1 -0
- package/dist/TokensController.js +15 -636
- package/dist/TokensController.js.map +1 -1
- package/dist/TokensController.mjs +16 -0
- package/dist/TokensController.mjs.map +1 -0
- package/dist/assetsUtil.js +41 -380
- package/dist/assetsUtil.js.map +1 -1
- package/dist/assetsUtil.mjs +42 -0
- package/dist/assetsUtil.mjs.map +1 -0
- package/dist/chunk-23F5W3A2.mjs +157 -0
- package/dist/chunk-23F5W3A2.mjs.map +1 -0
- package/dist/chunk-27KOXCQK.mjs +1 -0
- package/dist/chunk-27KOXCQK.mjs.map +1 -0
- package/dist/chunk-3FMWI46J.mjs +361 -0
- package/dist/chunk-3FMWI46J.mjs.map +1 -0
- package/dist/chunk-3FOTFZIL.mjs +411 -0
- package/dist/chunk-3FOTFZIL.mjs.map +1 -0
- package/dist/chunk-4LY47RPI.mjs +139 -0
- package/dist/chunk-4LY47RPI.mjs.map +1 -0
- package/dist/chunk-5C7ZVZNM.mjs +267 -0
- package/dist/chunk-5C7ZVZNM.mjs.map +1 -0
- package/dist/chunk-64EHFYLM.mjs +245 -0
- package/dist/chunk-64EHFYLM.mjs.map +1 -0
- package/dist/chunk-6C2ZIK7A.js +245 -0
- package/dist/chunk-6C2ZIK7A.js.map +1 -0
- package/dist/chunk-6DTCHPBB.mjs +178 -0
- package/dist/chunk-6DTCHPBB.mjs.map +1 -0
- package/dist/chunk-74IP34EL.js +1177 -0
- package/dist/chunk-74IP34EL.js.map +1 -0
- package/dist/chunk-BZI7P3TD.js +115 -0
- package/dist/chunk-BZI7P3TD.js.map +1 -0
- package/dist/chunk-C4PQK53K.js +32 -0
- package/dist/chunk-C4PQK53K.js.map +1 -0
- package/dist/chunk-CP3HC7AQ.mjs +12 -0
- package/dist/chunk-CP3HC7AQ.mjs.map +1 -0
- package/dist/chunk-GHZX5DE4.js +215 -0
- package/dist/chunk-GHZX5DE4.js.map +1 -0
- package/dist/chunk-GMKIOYCB.js +361 -0
- package/dist/chunk-GMKIOYCB.js.map +1 -0
- package/dist/chunk-H6TOSWUM.js +157 -0
- package/dist/chunk-H6TOSWUM.js.map +1 -0
- package/dist/chunk-HWFBJFHS.js +284 -0
- package/dist/chunk-HWFBJFHS.js.map +1 -0
- package/dist/chunk-J2JQQNHN.mjs +115 -0
- package/dist/chunk-J2JQQNHN.mjs.map +1 -0
- package/dist/chunk-J6HPEQL3.mjs +704 -0
- package/dist/chunk-J6HPEQL3.mjs.map +1 -0
- package/dist/chunk-JC4WAN2J.mjs +1177 -0
- package/dist/chunk-JC4WAN2J.mjs.map +1 -0
- package/dist/chunk-LD4GC7OR.js +139 -0
- package/dist/chunk-LD4GC7OR.js.map +1 -0
- package/dist/chunk-NG2UOKDD.js +178 -0
- package/dist/chunk-NG2UOKDD.js.map +1 -0
- package/dist/chunk-NGIXA5M5.mjs +418 -0
- package/dist/chunk-NGIXA5M5.mjs.map +1 -0
- package/dist/chunk-NLNXQHAU.js +704 -0
- package/dist/chunk-NLNXQHAU.js.map +1 -0
- package/dist/chunk-OBUR4TXH.js +1 -0
- package/dist/chunk-OBUR4TXH.js.map +1 -0
- package/dist/chunk-PAJTKWEC.mjs +246 -0
- package/dist/chunk-PAJTKWEC.mjs.map +1 -0
- package/dist/chunk-PRIXT2R6.js +411 -0
- package/dist/chunk-PRIXT2R6.js.map +1 -0
- package/dist/chunk-PUFSYRJZ.js +74 -0
- package/dist/chunk-PUFSYRJZ.js.map +1 -0
- package/dist/chunk-QHRPRO5U.mjs +74 -0
- package/dist/chunk-QHRPRO5U.mjs.map +1 -0
- package/dist/chunk-QR4CX2JT.mjs +284 -0
- package/dist/chunk-QR4CX2JT.mjs.map +1 -0
- package/dist/chunk-RGHTNZQ6.js +12 -0
- package/dist/chunk-RGHTNZQ6.js.map +1 -0
- package/dist/chunk-TCO22VIO.mjs +32 -0
- package/dist/chunk-TCO22VIO.mjs.map +1 -0
- package/dist/chunk-TDKCVCGP.mjs +215 -0
- package/dist/chunk-TDKCVCGP.mjs.map +1 -0
- package/dist/chunk-UAH5YURZ.js +343 -0
- package/dist/chunk-UAH5YURZ.js.map +1 -0
- package/dist/chunk-V4ZO3F2S.js +246 -0
- package/dist/chunk-V4ZO3F2S.js.map +1 -0
- package/dist/chunk-VDJBJAUB.js +418 -0
- package/dist/chunk-VDJBJAUB.js.map +1 -0
- package/dist/chunk-WRQ7POD7.mjs +343 -0
- package/dist/chunk-WRQ7POD7.mjs.map +1 -0
- package/dist/chunk-XUI43LEZ.mjs +30 -0
- package/dist/chunk-XUI43LEZ.mjs.map +1 -0
- package/dist/chunk-YD3NRMFC.js +267 -0
- package/dist/chunk-YD3NRMFC.js.map +1 -0
- package/dist/chunk-Z4BLTVTB.js +30 -0
- package/dist/chunk-Z4BLTVTB.js.map +1 -0
- package/dist/constants.js +7 -9
- package/dist/constants.js.map +1 -1
- package/dist/constants.mjs +8 -0
- package/dist/constants.mjs.map +1 -0
- package/dist/crypto-compare.js +7 -66
- package/dist/crypto-compare.js.map +1 -1
- package/dist/crypto-compare.mjs +8 -0
- package/dist/crypto-compare.mjs.map +1 -0
- package/dist/index.js +72 -36
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +73 -0
- package/dist/index.mjs.map +1 -0
- package/dist/token-prices-service/abstract-token-prices-service.js +1 -3
- package/dist/token-prices-service/abstract-token-prices-service.js.map +1 -1
- package/dist/token-prices-service/abstract-token-prices-service.mjs +1 -0
- package/dist/token-prices-service/abstract-token-prices-service.mjs.map +1 -0
- package/dist/token-prices-service/codefi-v2.js +11 -353
- package/dist/token-prices-service/codefi-v2.js.map +1 -1
- package/dist/token-prices-service/codefi-v2.mjs +12 -0
- package/dist/token-prices-service/codefi-v2.mjs.map +1 -0
- package/dist/token-prices-service/index.js +8 -5
- package/dist/token-prices-service/index.js.map +1 -1
- package/dist/token-prices-service/index.mjs +9 -0
- package/dist/token-prices-service/index.mjs.map +1 -0
- package/dist/token-service.js +14 -133
- package/dist/token-service.js.map +1 -1
- package/dist/token-service.mjs +15 -0
- package/dist/token-service.mjs.map +1 -0
- package/dist/tsconfig.build.tsbuildinfo +1 -0
- package/dist/types/AccountTrackerController.d.ts.map +1 -0
- package/dist/{AssetsContractController.d.ts → types/AssetsContractController.d.ts} +5 -6
- package/dist/types/AssetsContractController.d.ts.map +1 -0
- package/dist/types/CurrencyRateController.d.ts.map +1 -0
- package/dist/types/NftController.d.ts.map +1 -0
- package/dist/types/NftDetectionController.d.ts.map +1 -0
- package/dist/{Standards → types/Standards}/ERC20Standard.d.ts +1 -2
- package/dist/types/Standards/ERC20Standard.d.ts.map +1 -0
- package/dist/{Standards → types/Standards}/NftStandards/ERC1155/ERC1155Standard.d.ts +1 -2
- package/dist/types/Standards/NftStandards/ERC1155/ERC1155Standard.d.ts.map +1 -0
- package/dist/types/Standards/NftStandards/ERC721/ERC721Standard.d.ts.map +1 -0
- package/dist/{TokenBalancesController.d.ts → types/TokenBalancesController.d.ts} +12 -10
- package/dist/types/TokenBalancesController.d.ts.map +1 -0
- package/dist/types/TokenDetectionController.d.ts +111 -0
- package/dist/types/TokenDetectionController.d.ts.map +1 -0
- package/dist/types/TokenListController.d.ts.map +1 -0
- package/dist/types/TokenRatesController.d.ts.map +1 -0
- package/dist/{TokensController.d.ts → types/TokensController.d.ts} +26 -19
- package/dist/types/TokensController.d.ts.map +1 -0
- package/dist/{assetsUtil.d.ts → types/assetsUtil.d.ts} +1 -2
- package/dist/types/assetsUtil.d.ts.map +1 -0
- package/dist/types/constants.d.ts.map +1 -0
- package/dist/types/crypto-compare.d.ts.map +1 -0
- package/dist/{index.d.ts → types/index.d.ts} +6 -3
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/token-prices-service/abstract-token-prices-service.d.ts.map +1 -0
- package/dist/types/token-prices-service/codefi-v2.d.ts.map +1 -0
- package/dist/types/token-prices-service/index.d.ts.map +1 -0
- package/dist/types/token-service.d.ts.map +1 -0
- package/package.json +30 -13
- package/dist/AccountTrackerController.d.ts.map +0 -1
- package/dist/AssetsContractController.d.ts.map +0 -1
- package/dist/CurrencyRateController.d.ts.map +0 -1
- package/dist/NftController.d.ts.map +0 -1
- package/dist/NftDetectionController.d.ts.map +0 -1
- package/dist/Standards/ERC20Standard.d.ts.map +0 -1
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.d.ts.map +0 -1
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.d.ts.map +0 -1
- package/dist/Standards/standards-types.d.ts +0 -15
- package/dist/Standards/standards-types.d.ts.map +0 -1
- package/dist/Standards/standards-types.js +0 -3
- package/dist/Standards/standards-types.js.map +0 -1
- package/dist/TokenBalancesController.d.ts.map +0 -1
- package/dist/TokenDetectionController.d.ts +0 -88
- package/dist/TokenDetectionController.d.ts.map +0 -1
- package/dist/TokenListController.d.ts.map +0 -1
- package/dist/TokenRatesController.d.ts.map +0 -1
- package/dist/TokensController.d.ts.map +0 -1
- package/dist/assetsUtil.d.ts.map +0 -1
- package/dist/constants.d.ts.map +0 -1
- package/dist/crypto-compare.d.ts.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/token-prices-service/abstract-token-prices-service.d.ts.map +0 -1
- package/dist/token-prices-service/codefi-v2.d.ts.map +0 -1
- package/dist/token-prices-service/index.d.ts.map +0 -1
- package/dist/token-service.d.ts.map +0 -1
- /package/dist/{AccountTrackerController.d.ts → types/AccountTrackerController.d.ts} +0 -0
- /package/dist/{CurrencyRateController.d.ts → types/CurrencyRateController.d.ts} +0 -0
- /package/dist/{NftController.d.ts → types/NftController.d.ts} +0 -0
- /package/dist/{NftDetectionController.d.ts → types/NftDetectionController.d.ts} +0 -0
- /package/dist/{Standards → types/Standards}/NftStandards/ERC721/ERC721Standard.d.ts +0 -0
- /package/dist/{TokenListController.d.ts → types/TokenListController.d.ts} +0 -0
- /package/dist/{TokenRatesController.d.ts → types/TokenRatesController.d.ts} +0 -0
- /package/dist/{constants.d.ts → types/constants.d.ts} +0 -0
- /package/dist/{crypto-compare.d.ts → types/crypto-compare.d.ts} +0 -0
- /package/dist/{token-prices-service → types/token-prices-service}/abstract-token-prices-service.d.ts +0 -0
- /package/dist/{token-prices-service → types/token-prices-service}/codefi-v2.d.ts +0 -0
- /package/dist/{token-prices-service → types/token-prices-service}/index.d.ts +0 -0
- /package/dist/{token-service.d.ts → types/token-service.d.ts} +0 -0
|
@@ -0,0 +1,411 @@
|
|
|
1
|
+
import {
|
|
2
|
+
isTokenDetectionSupportedForNetwork
|
|
3
|
+
} from "./chunk-5C7ZVZNM.mjs";
|
|
4
|
+
import {
|
|
5
|
+
__privateAdd,
|
|
6
|
+
__privateGet,
|
|
7
|
+
__privateMethod,
|
|
8
|
+
__privateSet
|
|
9
|
+
} from "./chunk-XUI43LEZ.mjs";
|
|
10
|
+
|
|
11
|
+
// src/TokenDetectionController.ts
|
|
12
|
+
import contractMap from "@metamask/contract-metadata";
|
|
13
|
+
import { ChainId, safelyExecute } from "@metamask/controller-utils";
|
|
14
|
+
import { StaticIntervalPollingController } from "@metamask/polling-controller";
|
|
15
|
+
var DEFAULT_INTERVAL = 18e4;
|
|
16
|
+
function isEqualCaseInsensitive(value1, value2) {
|
|
17
|
+
if (typeof value1 !== "string" || typeof value2 !== "string") {
|
|
18
|
+
return false;
|
|
19
|
+
}
|
|
20
|
+
return value1.toLowerCase() === value2.toLowerCase();
|
|
21
|
+
}
|
|
22
|
+
var STATIC_MAINNET_TOKEN_LIST = Object.entries(
|
|
23
|
+
contractMap
|
|
24
|
+
).reduce((acc, [base, contract]) => {
|
|
25
|
+
const { logo, erc20, erc721, ...tokenMetadata } = contract;
|
|
26
|
+
return {
|
|
27
|
+
...acc,
|
|
28
|
+
[base.toLowerCase()]: {
|
|
29
|
+
...tokenMetadata,
|
|
30
|
+
address: base.toLowerCase(),
|
|
31
|
+
iconUrl: `images/contract/${logo}`,
|
|
32
|
+
aggregators: []
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
}, {});
|
|
36
|
+
var controllerName = "TokenDetectionController";
|
|
37
|
+
var _intervalId, _selectedAddress, _networkClientId, _tokenList, _disabled, _isUnlocked, _isDetectionEnabledFromPreferences, _isDetectionEnabledForNetwork, _getBalancesInSingleCall, _trackMetaMetricsEvent, _registerEventListeners, registerEventListeners_fn, _stopPolling, stopPolling_fn, _startPolling, startPolling_fn, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn, _restartTokenDetection, restartTokenDetection_fn, _getSlicesOfTokensToDetect, getSlicesOfTokensToDetect_fn, _addDetectedTokens, addDetectedTokens_fn;
|
|
38
|
+
var TokenDetectionController = class extends StaticIntervalPollingController {
|
|
39
|
+
/**
|
|
40
|
+
* Creates a TokenDetectionController instance.
|
|
41
|
+
*
|
|
42
|
+
* @param options - The controller options.
|
|
43
|
+
* @param options.messenger - The controller messaging system.
|
|
44
|
+
* @param options.disabled - If set to true, all network requests are blocked.
|
|
45
|
+
* @param options.interval - Polling interval used to fetch new token rates
|
|
46
|
+
* @param options.selectedAddress - Vault selected address
|
|
47
|
+
* @param options.getBalancesInSingleCall - Gets the balances of a list of tokens for the given address.
|
|
48
|
+
* @param options.trackMetaMetricsEvent - Sets options for MetaMetrics event tracking.
|
|
49
|
+
*/
|
|
50
|
+
constructor({
|
|
51
|
+
selectedAddress,
|
|
52
|
+
interval = DEFAULT_INTERVAL,
|
|
53
|
+
disabled = true,
|
|
54
|
+
getBalancesInSingleCall,
|
|
55
|
+
trackMetaMetricsEvent,
|
|
56
|
+
messenger
|
|
57
|
+
}) {
|
|
58
|
+
super({
|
|
59
|
+
name: controllerName,
|
|
60
|
+
messenger,
|
|
61
|
+
state: {},
|
|
62
|
+
metadata: {}
|
|
63
|
+
});
|
|
64
|
+
/**
|
|
65
|
+
* Constructor helper for registering this controller's messaging system subscriptions to controller events.
|
|
66
|
+
*/
|
|
67
|
+
__privateAdd(this, _registerEventListeners);
|
|
68
|
+
__privateAdd(this, _stopPolling);
|
|
69
|
+
/**
|
|
70
|
+
* Starts a new polling interval.
|
|
71
|
+
*/
|
|
72
|
+
__privateAdd(this, _startPolling);
|
|
73
|
+
__privateAdd(this, _getCorrectChainIdAndNetworkClientId);
|
|
74
|
+
/**
|
|
75
|
+
* Restart token detection polling period and call detectNewTokens
|
|
76
|
+
* in case of address change or user session initialization.
|
|
77
|
+
*
|
|
78
|
+
* @param options - Options for restart token detection.
|
|
79
|
+
* @param options.selectedAddress - the selectedAddress against which to detect for token balances
|
|
80
|
+
* @param options.networkClientId - The ID of the network client to use.
|
|
81
|
+
*/
|
|
82
|
+
__privateAdd(this, _restartTokenDetection);
|
|
83
|
+
__privateAdd(this, _getSlicesOfTokensToDetect);
|
|
84
|
+
__privateAdd(this, _addDetectedTokens);
|
|
85
|
+
__privateAdd(this, _intervalId, void 0);
|
|
86
|
+
__privateAdd(this, _selectedAddress, void 0);
|
|
87
|
+
__privateAdd(this, _networkClientId, void 0);
|
|
88
|
+
__privateAdd(this, _tokenList, {});
|
|
89
|
+
__privateAdd(this, _disabled, void 0);
|
|
90
|
+
__privateAdd(this, _isUnlocked, void 0);
|
|
91
|
+
__privateAdd(this, _isDetectionEnabledFromPreferences, void 0);
|
|
92
|
+
__privateAdd(this, _isDetectionEnabledForNetwork, void 0);
|
|
93
|
+
__privateAdd(this, _getBalancesInSingleCall, void 0);
|
|
94
|
+
__privateAdd(this, _trackMetaMetricsEvent, void 0);
|
|
95
|
+
__privateSet(this, _disabled, disabled);
|
|
96
|
+
this.setIntervalLength(interval);
|
|
97
|
+
__privateSet(this, _selectedAddress, selectedAddress ?? this.messagingSystem.call("AccountsController:getSelectedAccount").address);
|
|
98
|
+
const { chainId, networkClientId } = __privateMethod(this, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn).call(this);
|
|
99
|
+
__privateSet(this, _networkClientId, networkClientId);
|
|
100
|
+
const { useTokenDetection: defaultUseTokenDetection } = this.messagingSystem.call("PreferencesController:getState");
|
|
101
|
+
__privateSet(this, _isDetectionEnabledFromPreferences, defaultUseTokenDetection);
|
|
102
|
+
__privateSet(this, _isDetectionEnabledForNetwork, isTokenDetectionSupportedForNetwork(chainId));
|
|
103
|
+
__privateSet(this, _getBalancesInSingleCall, getBalancesInSingleCall);
|
|
104
|
+
__privateSet(this, _trackMetaMetricsEvent, trackMetaMetricsEvent);
|
|
105
|
+
const { isUnlocked } = this.messagingSystem.call(
|
|
106
|
+
"KeyringController:getState"
|
|
107
|
+
);
|
|
108
|
+
__privateSet(this, _isUnlocked, isUnlocked);
|
|
109
|
+
__privateMethod(this, _registerEventListeners, registerEventListeners_fn).call(this);
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Allows controller to make active and passive polling requests
|
|
113
|
+
*/
|
|
114
|
+
enable() {
|
|
115
|
+
__privateSet(this, _disabled, false);
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Blocks controller from making network calls
|
|
119
|
+
*/
|
|
120
|
+
disable() {
|
|
121
|
+
__privateSet(this, _disabled, true);
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Internal isActive state
|
|
125
|
+
* @type {boolean}
|
|
126
|
+
*/
|
|
127
|
+
get isActive() {
|
|
128
|
+
return !__privateGet(this, _disabled) && __privateGet(this, _isUnlocked);
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Start polling for detected tokens.
|
|
132
|
+
*/
|
|
133
|
+
async start() {
|
|
134
|
+
this.enable();
|
|
135
|
+
await __privateMethod(this, _startPolling, startPolling_fn).call(this);
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Stop polling for detected tokens.
|
|
139
|
+
*/
|
|
140
|
+
stop() {
|
|
141
|
+
this.disable();
|
|
142
|
+
__privateMethod(this, _stopPolling, stopPolling_fn).call(this);
|
|
143
|
+
}
|
|
144
|
+
async _executePoll(networkClientId, options) {
|
|
145
|
+
if (!this.isActive) {
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
await this.detectTokens({
|
|
149
|
+
networkClientId,
|
|
150
|
+
selectedAddress: options.address
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* For each token in the token list provided by the TokenListController, checks the token's balance for the selected account address on the active network.
|
|
155
|
+
* On mainnet, if token detection is disabled in preferences, ERC20 token auto detection will be triggered for each contract address in the legacy token list from the @metamask/contract-metadata repo.
|
|
156
|
+
*
|
|
157
|
+
* @param options - Options for token detection.
|
|
158
|
+
* @param options.networkClientId - The ID of the network client to use.
|
|
159
|
+
* @param options.selectedAddress - the selectedAddress against which to detect for token balances.
|
|
160
|
+
*/
|
|
161
|
+
async detectTokens({
|
|
162
|
+
networkClientId,
|
|
163
|
+
selectedAddress
|
|
164
|
+
} = {}) {
|
|
165
|
+
if (!this.isActive) {
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
const addressAgainstWhichToDetect = selectedAddress ?? __privateGet(this, _selectedAddress);
|
|
169
|
+
const { chainId, networkClientId: selectedNetworkClientId } = __privateMethod(this, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn).call(this, networkClientId);
|
|
170
|
+
const chainIdAgainstWhichToDetect = chainId;
|
|
171
|
+
const networkClientIdAgainstWhichToDetect = selectedNetworkClientId;
|
|
172
|
+
if (!isTokenDetectionSupportedForNetwork(chainIdAgainstWhichToDetect)) {
|
|
173
|
+
return;
|
|
174
|
+
}
|
|
175
|
+
if (!__privateGet(this, _isDetectionEnabledFromPreferences) && chainIdAgainstWhichToDetect !== ChainId.mainnet) {
|
|
176
|
+
return;
|
|
177
|
+
}
|
|
178
|
+
const isTokenDetectionInactiveInMainnet = !__privateGet(this, _isDetectionEnabledFromPreferences) && chainIdAgainstWhichToDetect === ChainId.mainnet;
|
|
179
|
+
const { tokensChainsCache } = this.messagingSystem.call(
|
|
180
|
+
"TokenListController:getState"
|
|
181
|
+
);
|
|
182
|
+
__privateSet(this, _tokenList, isTokenDetectionInactiveInMainnet ? STATIC_MAINNET_TOKEN_LIST : tokensChainsCache[chainIdAgainstWhichToDetect]?.data ?? {});
|
|
183
|
+
for (const tokensSlice of __privateMethod(this, _getSlicesOfTokensToDetect, getSlicesOfTokensToDetect_fn).call(this, {
|
|
184
|
+
chainId: chainIdAgainstWhichToDetect,
|
|
185
|
+
selectedAddress: addressAgainstWhichToDetect
|
|
186
|
+
})) {
|
|
187
|
+
await __privateMethod(this, _addDetectedTokens, addDetectedTokens_fn).call(this, {
|
|
188
|
+
tokensSlice,
|
|
189
|
+
selectedAddress: addressAgainstWhichToDetect,
|
|
190
|
+
networkClientId: networkClientIdAgainstWhichToDetect,
|
|
191
|
+
chainId: chainIdAgainstWhichToDetect
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
};
|
|
196
|
+
_intervalId = new WeakMap();
|
|
197
|
+
_selectedAddress = new WeakMap();
|
|
198
|
+
_networkClientId = new WeakMap();
|
|
199
|
+
_tokenList = new WeakMap();
|
|
200
|
+
_disabled = new WeakMap();
|
|
201
|
+
_isUnlocked = new WeakMap();
|
|
202
|
+
_isDetectionEnabledFromPreferences = new WeakMap();
|
|
203
|
+
_isDetectionEnabledForNetwork = new WeakMap();
|
|
204
|
+
_getBalancesInSingleCall = new WeakMap();
|
|
205
|
+
_trackMetaMetricsEvent = new WeakMap();
|
|
206
|
+
_registerEventListeners = new WeakSet();
|
|
207
|
+
registerEventListeners_fn = function() {
|
|
208
|
+
this.messagingSystem.subscribe("KeyringController:unlock", async () => {
|
|
209
|
+
__privateSet(this, _isUnlocked, true);
|
|
210
|
+
await __privateMethod(this, _restartTokenDetection, restartTokenDetection_fn).call(this);
|
|
211
|
+
});
|
|
212
|
+
this.messagingSystem.subscribe("KeyringController:lock", () => {
|
|
213
|
+
__privateSet(this, _isUnlocked, false);
|
|
214
|
+
__privateMethod(this, _stopPolling, stopPolling_fn).call(this);
|
|
215
|
+
});
|
|
216
|
+
this.messagingSystem.subscribe(
|
|
217
|
+
"TokenListController:stateChange",
|
|
218
|
+
async ({ tokenList }) => {
|
|
219
|
+
const hasTokens = Object.keys(tokenList).length;
|
|
220
|
+
if (hasTokens) {
|
|
221
|
+
await __privateMethod(this, _restartTokenDetection, restartTokenDetection_fn).call(this);
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
);
|
|
225
|
+
this.messagingSystem.subscribe(
|
|
226
|
+
"PreferencesController:stateChange",
|
|
227
|
+
async ({ selectedAddress: newSelectedAddress, useTokenDetection }) => {
|
|
228
|
+
const isSelectedAddressChanged = __privateGet(this, _selectedAddress) !== newSelectedAddress;
|
|
229
|
+
const isDetectionChangedFromPreferences = __privateGet(this, _isDetectionEnabledFromPreferences) !== useTokenDetection;
|
|
230
|
+
__privateSet(this, _selectedAddress, newSelectedAddress);
|
|
231
|
+
__privateSet(this, _isDetectionEnabledFromPreferences, useTokenDetection);
|
|
232
|
+
if (useTokenDetection && (isSelectedAddressChanged || isDetectionChangedFromPreferences)) {
|
|
233
|
+
await __privateMethod(this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
|
|
234
|
+
selectedAddress: __privateGet(this, _selectedAddress)
|
|
235
|
+
});
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
);
|
|
239
|
+
this.messagingSystem.subscribe(
|
|
240
|
+
"AccountsController:selectedAccountChange",
|
|
241
|
+
async ({ address: newSelectedAddress }) => {
|
|
242
|
+
const isSelectedAddressChanged = __privateGet(this, _selectedAddress) !== newSelectedAddress;
|
|
243
|
+
if (isSelectedAddressChanged && __privateGet(this, _isDetectionEnabledFromPreferences)) {
|
|
244
|
+
__privateSet(this, _selectedAddress, newSelectedAddress);
|
|
245
|
+
await __privateMethod(this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
|
|
246
|
+
selectedAddress: __privateGet(this, _selectedAddress)
|
|
247
|
+
});
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
);
|
|
251
|
+
this.messagingSystem.subscribe(
|
|
252
|
+
"NetworkController:networkDidChange",
|
|
253
|
+
async ({ selectedNetworkClientId }) => {
|
|
254
|
+
const isNetworkClientIdChanged = __privateGet(this, _networkClientId) !== selectedNetworkClientId;
|
|
255
|
+
const { chainId: newChainId } = __privateMethod(this, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn).call(this, selectedNetworkClientId);
|
|
256
|
+
__privateSet(this, _isDetectionEnabledForNetwork, isTokenDetectionSupportedForNetwork(newChainId));
|
|
257
|
+
if (isNetworkClientIdChanged && __privateGet(this, _isDetectionEnabledForNetwork)) {
|
|
258
|
+
__privateSet(this, _networkClientId, selectedNetworkClientId);
|
|
259
|
+
await __privateMethod(this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
|
|
260
|
+
networkClientId: __privateGet(this, _networkClientId)
|
|
261
|
+
});
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
);
|
|
265
|
+
};
|
|
266
|
+
_stopPolling = new WeakSet();
|
|
267
|
+
stopPolling_fn = function() {
|
|
268
|
+
if (__privateGet(this, _intervalId)) {
|
|
269
|
+
clearInterval(__privateGet(this, _intervalId));
|
|
270
|
+
}
|
|
271
|
+
};
|
|
272
|
+
_startPolling = new WeakSet();
|
|
273
|
+
startPolling_fn = async function() {
|
|
274
|
+
if (!this.isActive) {
|
|
275
|
+
return;
|
|
276
|
+
}
|
|
277
|
+
__privateMethod(this, _stopPolling, stopPolling_fn).call(this);
|
|
278
|
+
await this.detectTokens();
|
|
279
|
+
__privateSet(this, _intervalId, setInterval(async () => {
|
|
280
|
+
await this.detectTokens();
|
|
281
|
+
}, this.getIntervalLength()));
|
|
282
|
+
};
|
|
283
|
+
_getCorrectChainIdAndNetworkClientId = new WeakSet();
|
|
284
|
+
getCorrectChainIdAndNetworkClientId_fn = function(networkClientId) {
|
|
285
|
+
if (networkClientId) {
|
|
286
|
+
const networkConfiguration = this.messagingSystem.call(
|
|
287
|
+
"NetworkController:getNetworkConfigurationByNetworkClientId",
|
|
288
|
+
networkClientId
|
|
289
|
+
);
|
|
290
|
+
if (networkConfiguration) {
|
|
291
|
+
return {
|
|
292
|
+
chainId: networkConfiguration.chainId,
|
|
293
|
+
networkClientId
|
|
294
|
+
};
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
const { selectedNetworkClientId } = this.messagingSystem.call(
|
|
298
|
+
"NetworkController:getState"
|
|
299
|
+
);
|
|
300
|
+
const {
|
|
301
|
+
configuration: { chainId }
|
|
302
|
+
} = this.messagingSystem.call(
|
|
303
|
+
"NetworkController:getNetworkClientById",
|
|
304
|
+
selectedNetworkClientId
|
|
305
|
+
);
|
|
306
|
+
return {
|
|
307
|
+
chainId,
|
|
308
|
+
networkClientId: selectedNetworkClientId
|
|
309
|
+
};
|
|
310
|
+
};
|
|
311
|
+
_restartTokenDetection = new WeakSet();
|
|
312
|
+
restartTokenDetection_fn = async function({
|
|
313
|
+
selectedAddress,
|
|
314
|
+
networkClientId
|
|
315
|
+
} = {}) {
|
|
316
|
+
await this.detectTokens({
|
|
317
|
+
networkClientId,
|
|
318
|
+
selectedAddress
|
|
319
|
+
});
|
|
320
|
+
this.setIntervalLength(DEFAULT_INTERVAL);
|
|
321
|
+
};
|
|
322
|
+
_getSlicesOfTokensToDetect = new WeakSet();
|
|
323
|
+
getSlicesOfTokensToDetect_fn = function({
|
|
324
|
+
chainId,
|
|
325
|
+
selectedAddress
|
|
326
|
+
}) {
|
|
327
|
+
const { allTokens, allDetectedTokens, allIgnoredTokens } = this.messagingSystem.call("TokensController:getState");
|
|
328
|
+
const [tokensAddresses, detectedTokensAddresses, ignoredTokensAddresses] = [
|
|
329
|
+
allTokens,
|
|
330
|
+
allDetectedTokens,
|
|
331
|
+
allIgnoredTokens
|
|
332
|
+
].map(
|
|
333
|
+
(tokens) => (tokens[chainId]?.[selectedAddress] ?? []).map(
|
|
334
|
+
(value) => typeof value === "string" ? value : value.address
|
|
335
|
+
)
|
|
336
|
+
);
|
|
337
|
+
const tokensToDetect = [];
|
|
338
|
+
for (const tokenAddress of Object.keys(__privateGet(this, _tokenList))) {
|
|
339
|
+
if ([
|
|
340
|
+
tokensAddresses,
|
|
341
|
+
detectedTokensAddresses,
|
|
342
|
+
ignoredTokensAddresses
|
|
343
|
+
].every(
|
|
344
|
+
(addresses) => !addresses.find(
|
|
345
|
+
(address) => isEqualCaseInsensitive(address, tokenAddress)
|
|
346
|
+
)
|
|
347
|
+
)) {
|
|
348
|
+
tokensToDetect.push(tokenAddress);
|
|
349
|
+
}
|
|
350
|
+
}
|
|
351
|
+
const slicesOfTokensToDetect = [];
|
|
352
|
+
for (let i = 0, size = 1e3; i < tokensToDetect.length; i += size) {
|
|
353
|
+
slicesOfTokensToDetect.push(tokensToDetect.slice(i, i + size));
|
|
354
|
+
}
|
|
355
|
+
return slicesOfTokensToDetect;
|
|
356
|
+
};
|
|
357
|
+
_addDetectedTokens = new WeakSet();
|
|
358
|
+
addDetectedTokens_fn = async function({
|
|
359
|
+
tokensSlice,
|
|
360
|
+
selectedAddress,
|
|
361
|
+
networkClientId,
|
|
362
|
+
chainId
|
|
363
|
+
}) {
|
|
364
|
+
await safelyExecute(async () => {
|
|
365
|
+
const balances = await __privateGet(this, _getBalancesInSingleCall).call(this, selectedAddress, tokensSlice, networkClientId);
|
|
366
|
+
const tokensWithBalance = [];
|
|
367
|
+
const eventTokensDetails = [];
|
|
368
|
+
for (const nonZeroTokenAddress of Object.keys(balances)) {
|
|
369
|
+
const { decimals, symbol, aggregators, iconUrl, name } = __privateGet(this, _tokenList)[nonZeroTokenAddress];
|
|
370
|
+
eventTokensDetails.push(`${symbol} - ${nonZeroTokenAddress}`);
|
|
371
|
+
tokensWithBalance.push({
|
|
372
|
+
address: nonZeroTokenAddress,
|
|
373
|
+
decimals,
|
|
374
|
+
symbol,
|
|
375
|
+
aggregators,
|
|
376
|
+
image: iconUrl,
|
|
377
|
+
isERC721: false,
|
|
378
|
+
name
|
|
379
|
+
});
|
|
380
|
+
}
|
|
381
|
+
if (tokensWithBalance.length) {
|
|
382
|
+
__privateGet(this, _trackMetaMetricsEvent).call(this, {
|
|
383
|
+
event: "Token Detected",
|
|
384
|
+
category: "Wallet",
|
|
385
|
+
properties: {
|
|
386
|
+
tokens: eventTokensDetails,
|
|
387
|
+
token_standard: "ERC20",
|
|
388
|
+
asset_type: "TOKEN"
|
|
389
|
+
}
|
|
390
|
+
});
|
|
391
|
+
await this.messagingSystem.call(
|
|
392
|
+
"TokensController:addDetectedTokens",
|
|
393
|
+
tokensWithBalance,
|
|
394
|
+
{
|
|
395
|
+
selectedAddress,
|
|
396
|
+
chainId
|
|
397
|
+
}
|
|
398
|
+
);
|
|
399
|
+
}
|
|
400
|
+
});
|
|
401
|
+
};
|
|
402
|
+
var TokenDetectionController_default = TokenDetectionController;
|
|
403
|
+
|
|
404
|
+
export {
|
|
405
|
+
isEqualCaseInsensitive,
|
|
406
|
+
STATIC_MAINNET_TOKEN_LIST,
|
|
407
|
+
controllerName,
|
|
408
|
+
TokenDetectionController,
|
|
409
|
+
TokenDetectionController_default
|
|
410
|
+
};
|
|
411
|
+
//# sourceMappingURL=chunk-3FOTFZIL.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/TokenDetectionController.ts"],"sourcesContent":["import type {\n AccountsControllerGetSelectedAccountAction,\n AccountsControllerSelectedAccountChangeEvent,\n} from '@metamask/accounts-controller';\nimport type {\n RestrictedControllerMessenger,\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n} from '@metamask/base-controller';\nimport contractMap from '@metamask/contract-metadata';\nimport { ChainId, safelyExecute } from '@metamask/controller-utils';\nimport type {\n KeyringControllerGetStateAction,\n KeyringControllerLockEvent,\n KeyringControllerUnlockEvent,\n} from '@metamask/keyring-controller';\nimport type {\n NetworkClientId,\n NetworkControllerGetNetworkClientByIdAction,\n NetworkControllerGetNetworkConfigurationByNetworkClientId,\n NetworkControllerGetStateAction,\n NetworkControllerNetworkDidChangeEvent,\n} from '@metamask/network-controller';\nimport { StaticIntervalPollingController } from '@metamask/polling-controller';\nimport type {\n PreferencesControllerGetStateAction,\n PreferencesControllerStateChangeEvent,\n} from '@metamask/preferences-controller';\nimport type { Hex } from '@metamask/utils';\n\nimport type { AssetsContractController } from './AssetsContractController';\nimport { isTokenDetectionSupportedForNetwork } from './assetsUtil';\nimport type {\n GetTokenListState,\n TokenListMap,\n TokenListStateChange,\n} from './TokenListController';\nimport type { Token } from './TokenRatesController';\nimport type {\n TokensControllerAddDetectedTokensAction,\n TokensControllerGetStateAction,\n} from './TokensController';\n\nconst DEFAULT_INTERVAL = 180000;\n\n/**\n * Compare 2 given strings and return boolean\n * eg: \"foo\" and \"FOO\" => true\n * eg: \"foo\" and \"bar\" => false\n * eg: \"foo\" and 123 => false\n *\n * @param value1 - first string to compare\n * @param value2 - first string to compare\n * @returns true if 2 strings are identical when they are lowercase\n */\nexport function isEqualCaseInsensitive(\n value1: string,\n value2: string,\n): boolean {\n if (typeof value1 !== 'string' || typeof value2 !== 'string') {\n return false;\n }\n return value1.toLowerCase() === value2.toLowerCase();\n}\n\ntype LegacyToken = {\n name: string;\n logo: `${string}.svg`;\n symbol: string;\n decimals: number;\n erc20?: boolean;\n erc721?: boolean;\n};\n\ntype TokenDetectionMap = {\n [P in keyof TokenListMap]: Omit<TokenListMap[P], 'occurrences'>;\n};\n\nexport const STATIC_MAINNET_TOKEN_LIST = Object.entries<LegacyToken>(\n contractMap,\n).reduce<TokenDetectionMap>((acc, [base, contract]) => {\n const { logo, erc20, erc721, ...tokenMetadata } = contract;\n return {\n ...acc,\n [base.toLowerCase()]: {\n ...tokenMetadata,\n address: base.toLowerCase(),\n iconUrl: `images/contract/${logo}`,\n aggregators: [],\n },\n };\n}, {});\n\nexport const controllerName = 'TokenDetectionController';\n\nexport type TokenDetectionState = Record<never, never>;\n\nexport type TokenDetectionControllerGetStateAction = ControllerGetStateAction<\n typeof controllerName,\n TokenDetectionState\n>;\n\nexport type TokenDetectionControllerActions =\n TokenDetectionControllerGetStateAction;\n\nexport type AllowedActions =\n | AccountsControllerGetSelectedAccountAction\n | NetworkControllerGetNetworkClientByIdAction\n | NetworkControllerGetNetworkConfigurationByNetworkClientId\n | NetworkControllerGetStateAction\n | GetTokenListState\n | KeyringControllerGetStateAction\n | PreferencesControllerGetStateAction\n | TokensControllerGetStateAction\n | TokensControllerAddDetectedTokensAction;\n\nexport type TokenDetectionControllerStateChangeEvent =\n ControllerStateChangeEvent<typeof controllerName, TokenDetectionState>;\n\nexport type TokenDetectionControllerEvents =\n TokenDetectionControllerStateChangeEvent;\n\nexport type AllowedEvents =\n | AccountsControllerSelectedAccountChangeEvent\n | NetworkControllerNetworkDidChangeEvent\n | TokenListStateChange\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent\n | PreferencesControllerStateChangeEvent;\n\nexport type TokenDetectionControllerMessenger = RestrictedControllerMessenger<\n typeof controllerName,\n TokenDetectionControllerActions | AllowedActions,\n TokenDetectionControllerEvents | AllowedEvents,\n AllowedActions['type'],\n AllowedEvents['type']\n>;\n\n/**\n * Controller that passively polls on a set interval for Tokens auto detection\n * @property intervalId - Polling interval used to fetch new token rates\n * @property selectedAddress - Vault selected address\n * @property networkClientId - The network client ID of the current selected network\n * @property disabled - Boolean to track if network requests are blocked\n * @property isUnlocked - Boolean to track if the keyring state is unlocked\n * @property isDetectionEnabledFromPreferences - Boolean to track if detection is enabled from PreferencesController\n * @property isDetectionEnabledForNetwork - Boolean to track if detected is enabled for current network\n */\nexport class TokenDetectionController extends StaticIntervalPollingController<\n typeof controllerName,\n TokenDetectionState,\n TokenDetectionControllerMessenger\n> {\n #intervalId?: ReturnType<typeof setTimeout>;\n\n #selectedAddress: string;\n\n #networkClientId: NetworkClientId;\n\n #tokenList: TokenDetectionMap = {};\n\n #disabled: boolean;\n\n #isUnlocked: boolean;\n\n #isDetectionEnabledFromPreferences: boolean;\n\n #isDetectionEnabledForNetwork: boolean;\n\n readonly #getBalancesInSingleCall: AssetsContractController['getBalancesInSingleCall'];\n\n readonly #trackMetaMetricsEvent: (options: {\n event: string;\n category: string;\n properties: {\n tokens: string[];\n token_standard: string;\n asset_type: string;\n };\n }) => void;\n\n /**\n * Creates a TokenDetectionController instance.\n *\n * @param options - The controller options.\n * @param options.messenger - The controller messaging system.\n * @param options.disabled - If set to true, all network requests are blocked.\n * @param options.interval - Polling interval used to fetch new token rates\n * @param options.selectedAddress - Vault selected address\n * @param options.getBalancesInSingleCall - Gets the balances of a list of tokens for the given address.\n * @param options.trackMetaMetricsEvent - Sets options for MetaMetrics event tracking.\n */\n constructor({\n selectedAddress,\n interval = DEFAULT_INTERVAL,\n disabled = true,\n getBalancesInSingleCall,\n trackMetaMetricsEvent,\n messenger,\n }: {\n selectedAddress?: string;\n interval?: number;\n disabled?: boolean;\n getBalancesInSingleCall: AssetsContractController['getBalancesInSingleCall'];\n trackMetaMetricsEvent: (options: {\n event: string;\n category: string;\n properties: {\n tokens: string[];\n token_standard: string;\n asset_type: string;\n };\n }) => void;\n messenger: TokenDetectionControllerMessenger;\n }) {\n super({\n name: controllerName,\n messenger,\n state: {},\n metadata: {},\n });\n\n this.#disabled = disabled;\n this.setIntervalLength(interval);\n\n this.#selectedAddress =\n selectedAddress ??\n this.messagingSystem.call('AccountsController:getSelectedAccount')\n .address;\n\n const { chainId, networkClientId } =\n this.#getCorrectChainIdAndNetworkClientId();\n this.#networkClientId = networkClientId;\n\n const { useTokenDetection: defaultUseTokenDetection } =\n this.messagingSystem.call('PreferencesController:getState');\n this.#isDetectionEnabledFromPreferences = defaultUseTokenDetection;\n this.#isDetectionEnabledForNetwork =\n isTokenDetectionSupportedForNetwork(chainId);\n\n this.#getBalancesInSingleCall = getBalancesInSingleCall;\n\n this.#trackMetaMetricsEvent = trackMetaMetricsEvent;\n\n const { isUnlocked } = this.messagingSystem.call(\n 'KeyringController:getState',\n );\n this.#isUnlocked = isUnlocked;\n\n this.#registerEventListeners();\n }\n\n /**\n * Constructor helper for registering this controller's messaging system subscriptions to controller events.\n */\n #registerEventListeners() {\n this.messagingSystem.subscribe('KeyringController:unlock', async () => {\n this.#isUnlocked = true;\n await this.#restartTokenDetection();\n });\n\n this.messagingSystem.subscribe('KeyringController:lock', () => {\n this.#isUnlocked = false;\n this.#stopPolling();\n });\n\n this.messagingSystem.subscribe(\n 'TokenListController:stateChange',\n async ({ tokenList }) => {\n const hasTokens = Object.keys(tokenList).length;\n\n if (hasTokens) {\n await this.#restartTokenDetection();\n }\n },\n );\n\n this.messagingSystem.subscribe(\n 'PreferencesController:stateChange',\n async ({ selectedAddress: newSelectedAddress, useTokenDetection }) => {\n const isSelectedAddressChanged =\n this.#selectedAddress !== newSelectedAddress;\n const isDetectionChangedFromPreferences =\n this.#isDetectionEnabledFromPreferences !== useTokenDetection;\n\n this.#selectedAddress = newSelectedAddress;\n this.#isDetectionEnabledFromPreferences = useTokenDetection;\n\n if (\n useTokenDetection &&\n (isSelectedAddressChanged || isDetectionChangedFromPreferences)\n ) {\n await this.#restartTokenDetection({\n selectedAddress: this.#selectedAddress,\n });\n }\n },\n );\n\n this.messagingSystem.subscribe(\n 'AccountsController:selectedAccountChange',\n async ({ address: newSelectedAddress }) => {\n const isSelectedAddressChanged =\n this.#selectedAddress !== newSelectedAddress;\n if (\n isSelectedAddressChanged &&\n this.#isDetectionEnabledFromPreferences\n ) {\n this.#selectedAddress = newSelectedAddress;\n await this.#restartTokenDetection({\n selectedAddress: this.#selectedAddress,\n });\n }\n },\n );\n\n this.messagingSystem.subscribe(\n 'NetworkController:networkDidChange',\n async ({ selectedNetworkClientId }) => {\n const isNetworkClientIdChanged =\n this.#networkClientId !== selectedNetworkClientId;\n\n const { chainId: newChainId } =\n this.#getCorrectChainIdAndNetworkClientId(selectedNetworkClientId);\n this.#isDetectionEnabledForNetwork =\n isTokenDetectionSupportedForNetwork(newChainId);\n\n if (isNetworkClientIdChanged && this.#isDetectionEnabledForNetwork) {\n this.#networkClientId = selectedNetworkClientId;\n await this.#restartTokenDetection({\n networkClientId: this.#networkClientId,\n });\n }\n },\n );\n }\n\n /**\n * Allows controller to make active and passive polling requests\n */\n enable(): void {\n this.#disabled = false;\n }\n\n /**\n * Blocks controller from making network calls\n */\n disable(): void {\n this.#disabled = true;\n }\n\n /**\n * Internal isActive state\n * @type {boolean}\n */\n get isActive(): boolean {\n return !this.#disabled && this.#isUnlocked;\n }\n\n /**\n * Start polling for detected tokens.\n */\n async start(): Promise<void> {\n this.enable();\n await this.#startPolling();\n }\n\n /**\n * Stop polling for detected tokens.\n */\n stop(): void {\n this.disable();\n this.#stopPolling();\n }\n\n #stopPolling(): void {\n if (this.#intervalId) {\n clearInterval(this.#intervalId);\n }\n }\n\n /**\n * Starts a new polling interval.\n */\n async #startPolling(): Promise<void> {\n if (!this.isActive) {\n return;\n }\n this.#stopPolling();\n await this.detectTokens();\n this.#intervalId = setInterval(async () => {\n await this.detectTokens();\n }, this.getIntervalLength());\n }\n\n #getCorrectChainIdAndNetworkClientId(networkClientId?: NetworkClientId): {\n chainId: Hex;\n networkClientId: NetworkClientId;\n } {\n if (networkClientId) {\n const networkConfiguration = this.messagingSystem.call(\n 'NetworkController:getNetworkConfigurationByNetworkClientId',\n networkClientId,\n );\n if (networkConfiguration) {\n return {\n chainId: networkConfiguration.chainId,\n networkClientId,\n };\n }\n }\n const { selectedNetworkClientId } = this.messagingSystem.call(\n 'NetworkController:getState',\n );\n const {\n configuration: { chainId },\n } = this.messagingSystem.call(\n 'NetworkController:getNetworkClientById',\n selectedNetworkClientId,\n );\n return {\n chainId,\n networkClientId: selectedNetworkClientId,\n };\n }\n\n async _executePoll(\n networkClientId: NetworkClientId,\n options: { address: string },\n ): Promise<void> {\n if (!this.isActive) {\n return;\n }\n await this.detectTokens({\n networkClientId,\n selectedAddress: options.address,\n });\n }\n\n /**\n * Restart token detection polling period and call detectNewTokens\n * in case of address change or user session initialization.\n *\n * @param options - Options for restart token detection.\n * @param options.selectedAddress - the selectedAddress against which to detect for token balances\n * @param options.networkClientId - The ID of the network client to use.\n */\n async #restartTokenDetection({\n selectedAddress,\n networkClientId,\n }: {\n selectedAddress?: string;\n networkClientId?: NetworkClientId;\n } = {}): Promise<void> {\n await this.detectTokens({\n networkClientId,\n selectedAddress,\n });\n this.setIntervalLength(DEFAULT_INTERVAL);\n }\n\n /**\n * For each token in the token list provided by the TokenListController, checks the token's balance for the selected account address on the active network.\n * On mainnet, if token detection is disabled in preferences, ERC20 token auto detection will be triggered for each contract address in the legacy token list from the @metamask/contract-metadata repo.\n *\n * @param options - Options for token detection.\n * @param options.networkClientId - The ID of the network client to use.\n * @param options.selectedAddress - the selectedAddress against which to detect for token balances.\n */\n async detectTokens({\n networkClientId,\n selectedAddress,\n }: {\n networkClientId?: NetworkClientId;\n selectedAddress?: string;\n } = {}): Promise<void> {\n if (!this.isActive) {\n return;\n }\n\n const addressAgainstWhichToDetect =\n selectedAddress ?? this.#selectedAddress;\n const { chainId, networkClientId: selectedNetworkClientId } =\n this.#getCorrectChainIdAndNetworkClientId(networkClientId);\n const chainIdAgainstWhichToDetect = chainId;\n const networkClientIdAgainstWhichToDetect = selectedNetworkClientId;\n\n if (!isTokenDetectionSupportedForNetwork(chainIdAgainstWhichToDetect)) {\n return;\n }\n if (\n !this.#isDetectionEnabledFromPreferences &&\n chainIdAgainstWhichToDetect !== ChainId.mainnet\n ) {\n return;\n }\n const isTokenDetectionInactiveInMainnet =\n !this.#isDetectionEnabledFromPreferences &&\n chainIdAgainstWhichToDetect === ChainId.mainnet;\n const { tokensChainsCache } = this.messagingSystem.call(\n 'TokenListController:getState',\n );\n this.#tokenList = isTokenDetectionInactiveInMainnet\n ? STATIC_MAINNET_TOKEN_LIST\n : tokensChainsCache[chainIdAgainstWhichToDetect]?.data ?? {};\n\n for (const tokensSlice of this.#getSlicesOfTokensToDetect({\n chainId: chainIdAgainstWhichToDetect,\n selectedAddress: addressAgainstWhichToDetect,\n })) {\n await this.#addDetectedTokens({\n tokensSlice,\n selectedAddress: addressAgainstWhichToDetect,\n networkClientId: networkClientIdAgainstWhichToDetect,\n chainId: chainIdAgainstWhichToDetect,\n });\n }\n }\n\n #getSlicesOfTokensToDetect({\n chainId,\n selectedAddress,\n }: {\n chainId: Hex;\n selectedAddress: string;\n }): string[][] {\n const { allTokens, allDetectedTokens, allIgnoredTokens } =\n this.messagingSystem.call('TokensController:getState');\n const [tokensAddresses, detectedTokensAddresses, ignoredTokensAddresses] = [\n allTokens,\n allDetectedTokens,\n allIgnoredTokens,\n ].map((tokens) =>\n (tokens[chainId]?.[selectedAddress] ?? []).map((value) =>\n typeof value === 'string' ? value : value.address,\n ),\n );\n\n const tokensToDetect: string[] = [];\n for (const tokenAddress of Object.keys(this.#tokenList)) {\n if (\n [\n tokensAddresses,\n detectedTokensAddresses,\n ignoredTokensAddresses,\n ].every(\n (addresses) =>\n !addresses.find((address) =>\n isEqualCaseInsensitive(address, tokenAddress),\n ),\n )\n ) {\n tokensToDetect.push(tokenAddress);\n }\n }\n\n const slicesOfTokensToDetect = [];\n for (let i = 0, size = 1000; i < tokensToDetect.length; i += size) {\n slicesOfTokensToDetect.push(tokensToDetect.slice(i, i + size));\n }\n\n return slicesOfTokensToDetect;\n }\n\n async #addDetectedTokens({\n tokensSlice,\n selectedAddress,\n networkClientId,\n chainId,\n }: {\n tokensSlice: string[];\n selectedAddress: string;\n networkClientId: NetworkClientId;\n chainId: Hex;\n }): Promise<void> {\n await safelyExecute(async () => {\n const balances = await this.#getBalancesInSingleCall(\n selectedAddress,\n tokensSlice,\n networkClientId,\n );\n\n const tokensWithBalance: Token[] = [];\n const eventTokensDetails: string[] = [];\n for (const nonZeroTokenAddress of Object.keys(balances)) {\n const { decimals, symbol, aggregators, iconUrl, name } =\n this.#tokenList[nonZeroTokenAddress];\n eventTokensDetails.push(`${symbol} - ${nonZeroTokenAddress}`);\n tokensWithBalance.push({\n address: nonZeroTokenAddress,\n decimals,\n symbol,\n aggregators,\n image: iconUrl,\n isERC721: false,\n name,\n });\n }\n\n if (tokensWithBalance.length) {\n this.#trackMetaMetricsEvent({\n event: 'Token Detected',\n category: 'Wallet',\n properties: {\n tokens: eventTokensDetails,\n token_standard: 'ERC20',\n asset_type: 'TOKEN',\n },\n });\n\n await this.messagingSystem.call(\n 'TokensController:addDetectedTokens',\n tokensWithBalance,\n {\n selectedAddress,\n chainId,\n },\n );\n }\n });\n }\n}\n\nexport default TokenDetectionController;\n"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
import {
|
|
2
|
+
__privateAdd,
|
|
3
|
+
__privateGet,
|
|
4
|
+
__privateSet
|
|
5
|
+
} from "./chunk-XUI43LEZ.mjs";
|
|
6
|
+
|
|
7
|
+
// src/TokenBalancesController.ts
|
|
8
|
+
import {
|
|
9
|
+
BaseController
|
|
10
|
+
} from "@metamask/base-controller";
|
|
11
|
+
import { safelyExecute, toHex } from "@metamask/controller-utils";
|
|
12
|
+
var DEFAULT_INTERVAL = 18e4;
|
|
13
|
+
var controllerName = "TokenBalancesController";
|
|
14
|
+
var metadata = {
|
|
15
|
+
contractBalances: { persist: true, anonymous: false }
|
|
16
|
+
};
|
|
17
|
+
function getDefaultTokenBalancesState() {
|
|
18
|
+
return {
|
|
19
|
+
contractBalances: {}
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
var _handle, _getERC20BalanceOf, _interval, _tokens, _disabled;
|
|
23
|
+
var TokenBalancesController = class extends BaseController {
|
|
24
|
+
/**
|
|
25
|
+
* Construct a Token Balances Controller.
|
|
26
|
+
*
|
|
27
|
+
* @param options - The controller options.
|
|
28
|
+
* @param options.interval - Polling interval used to fetch new token balances.
|
|
29
|
+
* @param options.tokens - List of tokens to track balances for.
|
|
30
|
+
* @param options.disabled - If set to true, all tracked tokens contract balances updates are blocked.
|
|
31
|
+
* @param options.getERC20BalanceOf - Gets the balance of the given account at the given contract address.
|
|
32
|
+
* @param options.state - Initial state to set on this controller.
|
|
33
|
+
* @param options.messenger - The controller restricted messenger.
|
|
34
|
+
*/
|
|
35
|
+
constructor({
|
|
36
|
+
interval = DEFAULT_INTERVAL,
|
|
37
|
+
tokens = [],
|
|
38
|
+
disabled = false,
|
|
39
|
+
getERC20BalanceOf,
|
|
40
|
+
messenger,
|
|
41
|
+
state = {}
|
|
42
|
+
}) {
|
|
43
|
+
super({
|
|
44
|
+
name: controllerName,
|
|
45
|
+
metadata,
|
|
46
|
+
messenger,
|
|
47
|
+
state: {
|
|
48
|
+
...getDefaultTokenBalancesState(),
|
|
49
|
+
...state
|
|
50
|
+
}
|
|
51
|
+
});
|
|
52
|
+
__privateAdd(this, _handle, void 0);
|
|
53
|
+
__privateAdd(this, _getERC20BalanceOf, void 0);
|
|
54
|
+
__privateAdd(this, _interval, void 0);
|
|
55
|
+
__privateAdd(this, _tokens, void 0);
|
|
56
|
+
__privateAdd(this, _disabled, void 0);
|
|
57
|
+
__privateSet(this, _disabled, disabled);
|
|
58
|
+
__privateSet(this, _interval, interval);
|
|
59
|
+
__privateSet(this, _tokens, tokens);
|
|
60
|
+
this.messagingSystem.subscribe(
|
|
61
|
+
"TokensController:stateChange",
|
|
62
|
+
({ tokens: newTokens, detectedTokens }) => {
|
|
63
|
+
__privateSet(this, _tokens, [...newTokens, ...detectedTokens]);
|
|
64
|
+
this.updateBalances();
|
|
65
|
+
}
|
|
66
|
+
);
|
|
67
|
+
__privateSet(this, _getERC20BalanceOf, getERC20BalanceOf);
|
|
68
|
+
this.poll();
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Allows controller to update tracked tokens contract balances.
|
|
72
|
+
*/
|
|
73
|
+
enable() {
|
|
74
|
+
__privateSet(this, _disabled, false);
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Blocks controller from updating tracked tokens contract balances.
|
|
78
|
+
*/
|
|
79
|
+
disable() {
|
|
80
|
+
__privateSet(this, _disabled, true);
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Starts a new polling interval.
|
|
84
|
+
*
|
|
85
|
+
* @param interval - Polling interval used to fetch new token balances.
|
|
86
|
+
*/
|
|
87
|
+
async poll(interval) {
|
|
88
|
+
if (interval) {
|
|
89
|
+
__privateSet(this, _interval, interval);
|
|
90
|
+
}
|
|
91
|
+
if (__privateGet(this, _handle)) {
|
|
92
|
+
clearTimeout(__privateGet(this, _handle));
|
|
93
|
+
}
|
|
94
|
+
await safelyExecute(() => this.updateBalances());
|
|
95
|
+
__privateSet(this, _handle, setTimeout(() => {
|
|
96
|
+
this.poll(__privateGet(this, _interval));
|
|
97
|
+
}, __privateGet(this, _interval)));
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Updates balances for all tokens.
|
|
101
|
+
*/
|
|
102
|
+
async updateBalances() {
|
|
103
|
+
if (__privateGet(this, _disabled)) {
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
const newContractBalances = {};
|
|
107
|
+
for (const token of __privateGet(this, _tokens)) {
|
|
108
|
+
const { address } = token;
|
|
109
|
+
const { selectedAddress } = this.messagingSystem.call(
|
|
110
|
+
"PreferencesController:getState"
|
|
111
|
+
);
|
|
112
|
+
try {
|
|
113
|
+
newContractBalances[address] = toHex(
|
|
114
|
+
await __privateGet(this, _getERC20BalanceOf).call(this, address, selectedAddress)
|
|
115
|
+
);
|
|
116
|
+
token.balanceError = null;
|
|
117
|
+
} catch (error) {
|
|
118
|
+
newContractBalances[address] = toHex(0);
|
|
119
|
+
token.balanceError = error;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
this.update((state) => {
|
|
123
|
+
state.contractBalances = newContractBalances;
|
|
124
|
+
});
|
|
125
|
+
}
|
|
126
|
+
};
|
|
127
|
+
_handle = new WeakMap();
|
|
128
|
+
_getERC20BalanceOf = new WeakMap();
|
|
129
|
+
_interval = new WeakMap();
|
|
130
|
+
_tokens = new WeakMap();
|
|
131
|
+
_disabled = new WeakMap();
|
|
132
|
+
var TokenBalancesController_default = TokenBalancesController;
|
|
133
|
+
|
|
134
|
+
export {
|
|
135
|
+
getDefaultTokenBalancesState,
|
|
136
|
+
TokenBalancesController,
|
|
137
|
+
TokenBalancesController_default
|
|
138
|
+
};
|
|
139
|
+
//# sourceMappingURL=chunk-4LY47RPI.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/TokenBalancesController.ts"],"sourcesContent":["import {\n type RestrictedControllerMessenger,\n type ControllerGetStateAction,\n type ControllerStateChangeEvent,\n BaseController,\n} from '@metamask/base-controller';\nimport { safelyExecute, toHex } from '@metamask/controller-utils';\nimport type { PreferencesControllerGetStateAction } from '@metamask/preferences-controller';\n\nimport type { AssetsContractController } from './AssetsContractController';\nimport type { Token } from './TokenRatesController';\nimport type { TokensControllerStateChangeEvent } from './TokensController';\n\nconst DEFAULT_INTERVAL = 180000;\n\nconst controllerName = 'TokenBalancesController';\n\nconst metadata = {\n contractBalances: { persist: true, anonymous: false },\n};\n\n/**\n * Token balances controller options\n * @property interval - Polling interval used to fetch new token balances.\n * @property tokens - List of tokens to track balances for.\n * @property disabled - If set to true, all tracked tokens contract balances updates are blocked.\n * @property getERC20BalanceOf - Gets the balance of the given account at the given contract address.\n */\ntype TokenBalancesControllerOptions = {\n interval?: number;\n tokens?: Token[];\n disabled?: boolean;\n getERC20BalanceOf: AssetsContractController['getERC20BalanceOf'];\n messenger: TokenBalancesControllerMessenger;\n state?: Partial<TokenBalancesControllerState>;\n};\n\n/**\n * Represents a mapping of hash token contract addresses to their balances.\n */\ntype ContractBalances = Record<string, string>;\n\n/**\n * Token balances controller state\n * @property contractBalances - Hash of token contract addresses to balances\n */\nexport type TokenBalancesControllerState = {\n contractBalances: ContractBalances;\n};\n\nexport type TokenBalancesControllerGetStateAction = ControllerGetStateAction<\n typeof controllerName,\n TokenBalancesControllerState\n>;\n\nexport type TokenBalancesControllerActions =\n TokenBalancesControllerGetStateAction;\n\nexport type AllowedActions = PreferencesControllerGetStateAction;\n\nexport type TokenBalancesControllerStateChangeEvent =\n ControllerStateChangeEvent<\n typeof controllerName,\n TokenBalancesControllerState\n >;\n\nexport type TokenBalancesControllerEvents =\n TokenBalancesControllerStateChangeEvent;\n\nexport type AllowedEvents = TokensControllerStateChangeEvent;\n\nexport type TokenBalancesControllerMessenger = RestrictedControllerMessenger<\n typeof controllerName,\n TokenBalancesControllerActions | AllowedActions,\n TokenBalancesControllerEvents | AllowedEvents,\n AllowedActions['type'],\n AllowedEvents['type']\n>;\n\n/**\n * Get the default TokenBalancesController state.\n *\n * @returns The default TokenBalancesController state.\n */\nexport function getDefaultTokenBalancesState(): TokenBalancesControllerState {\n return {\n contractBalances: {},\n };\n}\n\n/**\n * Controller that passively polls on a set interval token balances\n * for tokens stored in the TokensController\n */\nexport class TokenBalancesController extends BaseController<\n typeof controllerName,\n TokenBalancesControllerState,\n TokenBalancesControllerMessenger\n> {\n #handle?: ReturnType<typeof setTimeout>;\n\n #getERC20BalanceOf: AssetsContractController['getERC20BalanceOf'];\n\n #interval: number;\n\n #tokens: Token[];\n\n #disabled: boolean;\n\n /**\n * Construct a Token Balances Controller.\n *\n * @param options - The controller options.\n * @param options.interval - Polling interval used to fetch new token balances.\n * @param options.tokens - List of tokens to track balances for.\n * @param options.disabled - If set to true, all tracked tokens contract balances updates are blocked.\n * @param options.getERC20BalanceOf - Gets the balance of the given account at the given contract address.\n * @param options.state - Initial state to set on this controller.\n * @param options.messenger - The controller restricted messenger.\n */\n constructor({\n interval = DEFAULT_INTERVAL,\n tokens = [],\n disabled = false,\n getERC20BalanceOf,\n messenger,\n state = {},\n }: TokenBalancesControllerOptions) {\n super({\n name: controllerName,\n metadata,\n messenger,\n state: {\n ...getDefaultTokenBalancesState(),\n ...state,\n },\n });\n\n this.#disabled = disabled;\n this.#interval = interval;\n this.#tokens = tokens;\n\n this.messagingSystem.subscribe(\n 'TokensController:stateChange',\n ({ tokens: newTokens, detectedTokens }) => {\n this.#tokens = [...newTokens, ...detectedTokens];\n this.updateBalances();\n },\n );\n\n this.#getERC20BalanceOf = getERC20BalanceOf;\n\n this.poll();\n }\n\n /**\n * Allows controller to update tracked tokens contract balances.\n */\n enable() {\n this.#disabled = false;\n }\n\n /**\n * Blocks controller from updating tracked tokens contract balances.\n */\n disable() {\n this.#disabled = true;\n }\n\n /**\n * Starts a new polling interval.\n *\n * @param interval - Polling interval used to fetch new token balances.\n */\n async poll(interval?: number): Promise<void> {\n if (interval) {\n this.#interval = interval;\n }\n\n if (this.#handle) {\n clearTimeout(this.#handle);\n }\n\n await safelyExecute(() => this.updateBalances());\n\n this.#handle = setTimeout(() => {\n this.poll(this.#interval);\n }, this.#interval);\n }\n\n /**\n * Updates balances for all tokens.\n */\n async updateBalances() {\n if (this.#disabled) {\n return;\n }\n\n const newContractBalances: ContractBalances = {};\n for (const token of this.#tokens) {\n const { address } = token;\n const { selectedAddress } = this.messagingSystem.call(\n 'PreferencesController:getState',\n );\n try {\n newContractBalances[address] = toHex(\n await this.#getERC20BalanceOf(address, selectedAddress),\n );\n token.balanceError = null;\n } catch (error) {\n newContractBalances[address] = toHex(0);\n token.balanceError = error;\n }\n }\n\n this.update((state) => {\n state.contractBalances = newContractBalances;\n });\n }\n}\n\nexport default TokenBalancesController;\n"],"mappings":";;;;;;;AAAA;AAAA,EAIE;AAAA,OACK;AACP,SAAS,eAAe,aAAa;AAOrC,IAAM,mBAAmB;AAEzB,IAAM,iBAAiB;AAEvB,IAAM,WAAW;AAAA,EACf,kBAAkB,EAAE,SAAS,MAAM,WAAW,MAAM;AACtD;AAiEO,SAAS,+BAA6D;AAC3E,SAAO;AAAA,IACL,kBAAkB,CAAC;AAAA,EACrB;AACF;AAxFA;AA8FO,IAAM,0BAAN,cAAsC,eAI3C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsBA,YAAY;AAAA,IACV,WAAW;AAAA,IACX,SAAS,CAAC;AAAA,IACV,WAAW;AAAA,IACX;AAAA,IACA;AAAA,IACA,QAAQ,CAAC;AAAA,EACX,GAAmC;AACjC,UAAM;AAAA,MACJ,MAAM;AAAA,MACN;AAAA,MACA;AAAA,MACA,OAAO;AAAA,QACL,GAAG,6BAA6B;AAAA,QAChC,GAAG;AAAA,MACL;AAAA,IACF,CAAC;AArCH;AAEA;AAEA;AAEA;AAEA;AA+BE,uBAAK,WAAY;AACjB,uBAAK,WAAY;AACjB,uBAAK,SAAU;AAEf,SAAK,gBAAgB;AAAA,MACnB;AAAA,MACA,CAAC,EAAE,QAAQ,WAAW,eAAe,MAAM;AACzC,2BAAK,SAAU,CAAC,GAAG,WAAW,GAAG,cAAc;AAC/C,aAAK,eAAe;AAAA,MACtB;AAAA,IACF;AAEA,uBAAK,oBAAqB;AAE1B,SAAK,KAAK;AAAA,EACZ;AAAA;AAAA;AAAA;AAAA,EAKA,SAAS;AACP,uBAAK,WAAY;AAAA,EACnB;AAAA;AAAA;AAAA;AAAA,EAKA,UAAU;AACR,uBAAK,WAAY;AAAA,EACnB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,KAAK,UAAkC;AAC3C,QAAI,UAAU;AACZ,yBAAK,WAAY;AAAA,IACnB;AAEA,QAAI,mBAAK,UAAS;AAChB,mBAAa,mBAAK,QAAO;AAAA,IAC3B;AAEA,UAAM,cAAc,MAAM,KAAK,eAAe,CAAC;AAE/C,uBAAK,SAAU,WAAW,MAAM;AAC9B,WAAK,KAAK,mBAAK,UAAS;AAAA,IAC1B,GAAG,mBAAK,UAAS;AAAA,EACnB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,iBAAiB;AACrB,QAAI,mBAAK,YAAW;AAClB;AAAA,IACF;AAEA,UAAM,sBAAwC,CAAC;AAC/C,eAAW,SAAS,mBAAK,UAAS;AAChC,YAAM,EAAE,QAAQ,IAAI;AACpB,YAAM,EAAE,gBAAgB,IAAI,KAAK,gBAAgB;AAAA,QAC/C;AAAA,MACF;AACA,UAAI;AACF,4BAAoB,OAAO,IAAI;AAAA,UAC7B,MAAM,mBAAK,oBAAL,WAAwB,SAAS;AAAA,QACzC;AACA,cAAM,eAAe;AAAA,MACvB,SAAS,OAAO;AACd,4BAAoB,OAAO,IAAI,MAAM,CAAC;AACtC,cAAM,eAAe;AAAA,MACvB;AAAA,IACF;AAEA,SAAK,OAAO,CAAC,UAAU;AACrB,YAAM,mBAAmB;AAAA,IAC3B,CAAC;AAAA,EACH;AACF;AAxHE;AAEA;AAEA;AAEA;AAEA;AAkHF,IAAO,kCAAQ;","names":[]}
|