forked from dcgtc/dgrants
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathwallet.ts
262 lines (234 loc) · 11.3 KB
/
wallet.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
/**
* @dev Information about the user's wallet, network, etc. are stored and managed here
*
* @dev State is handled in reusable components, where each component is its own self-contained file consisting of
* one function defined used the composition API. Since we want the wallet state to be shared between all instances
* when this file is imported, we defined state outside of the function definition.
*
* @dev When assigning ethers objects as refs, we must wrap the object in `markRaw` for assignment. This is not required
* with Vue 2's reactivity system based on Object.defineProperty, but is required for Vue 3's reactivity system based
* on ES6 proxies. The Vue 3 reactivity system does not work well with non-configurable, non-writable properties on
* objects, and many ethers classes, such as providers and networks, use non-configurable or non-writable properties.
* Therefore we wrap the object in `markRaw` to prevent it from being converted to a Proxy. If you do not do this,
* you'll see errors like this when using ethers objects as refs:
* Uncaught (in promise) TypeError: 'get' on proxy: property '_network' is a read-only and non-configurable data
* property on the proxy target but the proxy did not return its actual value (expected '#<Object>' but got
* '[object Object]')
* Read more here:
* - https://stackoverflow.com/questions/65693108/threejs-component-working-in-vuejs-2-but-not-3
* - https://github.com/vuejs/vue-next/issues/3024
* - https://v3.vuejs.org/api/basic-reactivity.html#markraw
*/
import { computed, ref, markRaw } from 'vue';
import useDataStore from 'src/store/data';
import useSettingsStore from 'src/store/settings';
import { ALL_CHAIN_INFO, CHAIN_INFO, DGRANTS_CHAIN_ID, GRANT_REGISTRY_ADDRESS, GRANT_ROUND_MANAGER_ADDRESS, MULTICALL_ADDRESS, DEFAULT_PROVIDER, SupportedChainId } from 'src/utils/chains'; // prettier-ignore
import { BigNumber, Contract, hexStripZeros, JsonRpcProvider, JsonRpcSigner, Network, Web3Provider } from 'src/utils/ethers'; // prettier-ignore
import { formatAddress } from 'src/utils/utils';
import Onboard from 'bnc-onboard';
import { API as OnboardAPI } from 'bnc-onboard/dist/src/interfaces';
import { getAddress } from 'src/utils/ethers';
import { GRANT_REGISTRY_ABI, MULTICALL_ABI } from 'src/utils/constants';
import { GrantRegistry, GrantRoundManager } from '@dgrants/contracts';
const { startPolling } = useDataStore();
const { setLastWallet, clearLastWallet } = useSettingsStore();
const defaultChainId = SupportedChainId.MAINNET;
// State variables
let onboard: OnboardAPI; // instance of Blocknative's onboard.js library
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const rawProvider = ref<any>(); // raw provider from the user's wallet, e.g. EIP-1193 provider
const provider = ref<Web3Provider | JsonRpcProvider>(markRaw(DEFAULT_PROVIDER)); // ethers provider
const signer = ref<JsonRpcSigner>(); // ethers signer
const userAddress = ref<string>(); // user's wallet address
const userEns = ref<string | null>(); // user's ENS name
const network = ref<Network>(); // connected network, derived from provider
// Reset state when, e.g.user switches wallets. Provider/signer are automatically updated by ethers so are not cleared
function resetState() {
userAddress.value = undefined;
network.value = undefined;
}
// Settings
const rpcUrlsByChain: Record<string, string> = {};
Object.entries(ALL_CHAIN_INFO).forEach(([chainId, chainData]) => (rpcUrlsByChain[String(chainId)] = chainData.rpcUrl));
const walletChecks = [{ checkName: 'connect' }];
const wallets = [
{ walletName: 'metamask', preferred: true },
{ walletName: 'walletConnect', rpc: rpcUrlsByChain, preferred: true },
];
export default function useWalletStore() {
// ------------------------------------------------ Wallet Connection ------------------------------------------------
/**
* @notice Initialize the onboard.js module
*/
function initializeOnboard() {
onboard = Onboard({
dappId: import.meta.env.VITE_BLOCKNATIVE_API_KEY,
networkId: 1,
walletSelect: { wallets },
walletCheck: walletChecks,
subscriptions: {
// On wallet connection, save wallet in local storage and set provider
wallet: (wallet) => {
setProvider(wallet.provider);
if (wallet.name) setLastWallet(wallet.name);
},
// On address or network change, re-run configureProvider
address: async (address) => {
if (userAddress.value && userAddress.value !== getAddress(address)) await configureProvider();
},
network: async (chainId) => {
if (network.value?.chainId && network.value.chainId !== chainId) await configureProvider();
},
},
});
}
/**
* @notice Prompt user to connect wallet, or attempt to connect to wallet specified by `name`
* @param name Wallet name to connect, or undefined to prompt user to select a wallet
*/
async function connectWallet(name: string | undefined | MouseEvent = undefined) {
// If user already connected wallet, return
if (userAddress.value) return;
// If input type is MouseEvent, this method was ran from clicking a DOM element, so set name to undefined
if (name && typeof name !== 'string' && 'pageX' in name) name = undefined;
// Otherwise, prompt them for connection / wallet change
if (!onboard) initializeOnboard(); // instantiate Onboard instance
onboard.walletReset(); // clear existing wallet selection
await onboard.walletSelect(name); // wait for user to select wallet
await onboard.walletCheck(); // run any specified checks
await configureProvider(); // load info based on user's address
}
/**
* @notice Disconnect the current user wallet
*/
function disconnectWallet() {
// If user doesn't have a connected wallet, return
if (!userAddress.value) return;
resetState();
onboard.walletReset();
clearLastWallet();
}
/**
* @notice Change wallet
*/
async function changeWallet() {
const prevWallet = onboard.getState().wallet;
try {
await onboard.walletSelect();
await onboard.walletCheck();
await configureProvider();
} catch (error) {
if (prevWallet.name) {
// changing wallet failed. Restoring previous wallet
await onboard.walletSelect(prevWallet.name);
} else {
// this shouldn't happen because to change a wallet you need a previous wallet
// defaulting to disconnecting if we reach this point
disconnectWallet();
}
}
}
// ----------------------------------------------------- Actions -----------------------------------------------------
// When user connects their wallet, we call this method to update the provider
/* eslint-disable @typescript-eslint/no-explicit-any */
function setProvider(p: any) {
rawProvider.value = p;
}
/* eslint-enable @typescript-eslint/no-explicit-any */
// Any actions or data to fetch dependent on user's wallet are done here
async function configureProvider() {
// Set network/wallet properties
if (!rawProvider.value) return;
const _provider = new Web3Provider(rawProvider.value);
const _signer = _provider.getSigner();
// Get user and network information
const [_userAddress, _network] = await Promise.all([
_signer.getAddress(), // get user's address
_provider.getNetwork(), // get information on the connected network
]);
// If nothing has changed, no need to continue configuring
if (_userAddress === userAddress.value && _network.chainId === network.value?.chainId) return;
// Clear state
resetState();
// Get ENS name if user is connected to mainnet
const chainId = _provider.network.chainId; // must be done after the .getNetwork() call
const _userEns = chainId === 1 ? await _provider.lookupAddress(_userAddress) : null;
// Now we save the user's info to the store. We don't do this earlier because the UI is reactive based on these
// parameters, and we want to ensure this method completed successfully before updating the UI
if (chainId === DGRANTS_CHAIN_ID) provider.value = markRaw(_provider); // ensure we always read the correct data from DGRANTS_CHAIN_ID
signer.value = _signer;
userAddress.value = _userAddress;
userEns.value = _userEns;
network.value = markRaw(_network);
// Start polling for data
startPolling();
}
/**
* @notice Prompts user to switch networks to this applications required network
*/
async function switchNetwork() {
// Chain ID must be a 0x-prefixed hex string without paddingL https://eth.wiki/json-rpc/API
const chainId = hexStripZeros(BigNumber.from(DGRANTS_CHAIN_ID).toHexString());
// First try switching chains
try {
await rawProvider.value?.send('wallet_switchEthereumChain', [{ chainId }]);
} catch (err) {
const { code } = err as { code: number };
// This error code indicates that the chain has not been added to MetaMask: https://docs.metamask.io/guide/rpc-api.html#other-rpc-methods
// In this case, add the chain for the user
if (code === 4902) {
await rawProvider.value?.send('wallet_addEthereumChain', [
{
chainId,
chainName: CHAIN_INFO.label,
nativeCurrency: CHAIN_INFO.nativeCurrency,
rpcUrls: [CHAIN_INFO.rpcUrl],
blockExplorerUrls: [CHAIN_INFO.explorer],
},
]);
} else {
throw err;
}
}
}
// ----------------------------------------------------- Getters -----------------------------------------------------
// Default to mainnet for all network-based getters
const chainId = computed(() => (network.value?.chainId || defaultChainId) as SupportedChainId);
const contractProvider = computed(() => (chainId.value === DGRANTS_CHAIN_ID ? signer.value : DEFAULT_PROVIDER));
const grantRegistry = computed(() => {
return <GrantRegistry>new Contract(GRANT_REGISTRY_ADDRESS, GRANT_REGISTRY_ABI, contractProvider.value);
});
const grantRoundManager = computed(() => {
return <GrantRoundManager>(
new Contract(GRANT_ROUND_MANAGER_ADDRESS, CHAIN_INFO.grantRoundManagerAbi, contractProvider.value)
);
});
const multicall = computed(() => new Contract(MULTICALL_ADDRESS, MULTICALL_ABI, contractProvider.value));
const isSupportedNetwork = computed(
() => (network.value ? DGRANTS_CHAIN_ID === network.value.chainId : true) // assume valid if we have no network information
);
const isCorrectNetwork = computed(() => network.value && DGRANTS_CHAIN_ID === network.value.chainId);
// ----------------------------------------------------- Exports -----------------------------------------------------
// Define parts of the store to expose. Only expose computed properties or methods to avoid direct mutation of state
return {
// Methods
changeWallet,
configureProvider,
connectWallet,
disconnectWallet,
setProvider,
switchNetwork,
// Properties
chainId,
grantRegistry,
grantRoundManager,
isCorrectNetwork,
isSupportedNetwork,
multicall,
network: computed(() => network.value),
provider: computed(() => provider.value),
signer: computed(() => signer.value),
userAddress: computed(() => userAddress.value),
userDisplayName: computed(() => userEns.value || formatAddress(userAddress.value || '')),
};
}