@wagmi/connectors 4.0.0-alpha.5 → 4.0.0-alpha.7

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.
@@ -1,2 +1,2 @@
1
- export const version = '4.0.0-alpha.5';
1
+ export const version = '4.0.0-alpha.7';
2
2
  //# sourceMappingURL=version.js.map
@@ -1,6 +1,7 @@
1
1
  export { type InjectedParameters, injected, type MockParameters, mock, } from '@wagmi/core/internal';
2
2
  export { type CoinbaseWalletParameters, coinbaseWallet, } from '../coinbaseWallet.js';
3
3
  export { type LedgerParameters, ledger } from '../ledger.js';
4
+ export { type MetaMaskParameters, metaMask } from '../metaMask.js';
4
5
  export { type SafeParameters, safe } from '../safe.js';
5
6
  export { type WalletConnectParameters, walletConnect, } from '../walletConnect.js';
6
7
  export { version } from '../version.js';
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/exports/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,KAAK,kBAAkB,EACvB,QAAQ,EACR,KAAK,cAAc,EACnB,IAAI,GACL,MAAM,sBAAsB,CAAA;AAE7B,OAAO,EACL,KAAK,wBAAwB,EAC7B,cAAc,GACf,MAAM,sBAAsB,CAAA;AAE7B,OAAO,EAAE,KAAK,gBAAgB,EAAE,MAAM,EAAE,MAAM,cAAc,CAAA;AAE5D,OAAO,EAAE,KAAK,cAAc,EAAE,IAAI,EAAE,MAAM,YAAY,CAAA;AAEtD,OAAO,EACL,KAAK,uBAAuB,EAC5B,aAAa,GACd,MAAM,qBAAqB,CAAA;AAE5B,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAA"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/exports/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,KAAK,kBAAkB,EACvB,QAAQ,EACR,KAAK,cAAc,EACnB,IAAI,GACL,MAAM,sBAAsB,CAAA;AAE7B,OAAO,EACL,KAAK,wBAAwB,EAC7B,cAAc,GACf,MAAM,sBAAsB,CAAA;AAE7B,OAAO,EAAE,KAAK,gBAAgB,EAAE,MAAM,EAAE,MAAM,cAAc,CAAA;AAE5D,OAAO,EAAE,KAAK,kBAAkB,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAA;AAElE,OAAO,EAAE,KAAK,cAAc,EAAE,IAAI,EAAE,MAAM,YAAY,CAAA;AAEtD,OAAO,EACL,KAAK,uBAAuB,EAC5B,aAAa,GACd,MAAM,qBAAqB,CAAA;AAE5B,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAA"}
@@ -0,0 +1,10 @@
1
+ import { type MetaMaskSDKOptions, SDKProvider } from '@metamask/sdk';
2
+ import type { Evaluate, ExactPartial, Omit } from '@wagmi/core/internal';
3
+ import { type ProviderConnectInfo } from 'viem';
4
+ export type MetaMaskParameters = Evaluate<ExactPartial<Omit<MetaMaskSDKOptions, 'checkInstallationImmediately' | 'checkInstallationOnAllCalls' | 'defaultReadOnlyChainId' | 'readonlyRPCMap'>>>;
5
+ export declare function metaMask(parameters?: MetaMaskParameters): import("@wagmi/core").CreateConnectorFn<SDKProvider, {
6
+ onConnect(connectInfo: ProviderConnectInfo): void;
7
+ }, {
8
+ 'metaMaskSDK.disconnected': true;
9
+ }>;
10
+ //# sourceMappingURL=metaMask.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"metaMask.d.ts","sourceRoot":"","sources":["../../src/metaMask.ts"],"names":[],"mappings":"AAAA,OAAO,EAGL,KAAK,kBAAkB,EACvB,WAAW,EACZ,MAAM,eAAe,CAAA;AAMtB,OAAO,KAAK,EAAE,QAAQ,EAAE,YAAY,EAAE,IAAI,EAAE,MAAM,sBAAsB,CAAA;AACxE,OAAO,EAEL,KAAK,mBAAmB,EASzB,MAAM,MAAM,CAAA;AAEb,MAAM,MAAM,kBAAkB,GAAG,QAAQ,CACvC,YAAY,CACV,IAAI,CACF,kBAAkB,EAChB,8BAA8B,GAC9B,6BAA6B,GAC7B,wBAAwB,GACxB,gBAAgB,CACnB,CACF,CACF,CAAA;AAED,wBAAgB,QAAQ,CAAC,UAAU,GAAE,kBAAuB;2BAGjC,mBAAmB,GAAG,IAAI;;gCAEF,IAAI;GAiRtD"}
@@ -1,2 +1,2 @@
1
- export declare const version = "4.0.0-alpha.5";
1
+ export declare const version = "4.0.0-alpha.7";
2
2
  //# sourceMappingURL=version.d.ts.map
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@wagmi/connectors",
3
3
  "description": "Collection of connectors for Wagmi",
4
- "version": "4.0.0-alpha.5",
4
+ "version": "4.0.0-alpha.7",
5
5
  "license": "MIT",
6
6
  "repository": {
7
7
  "type": "git",
@@ -28,9 +28,9 @@
28
28
  "./package.json": "./package.json"
29
29
  },
30
30
  "peerDependencies": {
31
- "typescript": ">=5",
32
- "viem": "2.0.0-alpha.12",
33
- "@wagmi/core": "2.0.0-alpha.5"
31
+ "typescript": ">=5.0.4",
32
+ "viem": "2.0.0-alpha.13",
33
+ "@wagmi/core": "2.0.0-alpha.7"
34
34
  },
35
35
  "peerDependenciesMeta": {
36
36
  "typescript": {
@@ -40,6 +40,7 @@
40
40
  "dependencies": {
41
41
  "@coinbase/wallet-sdk": "3.9.0-canary.2",
42
42
  "@ledgerhq/connect-kit-loader": "^1.1.2",
43
+ "@metamask/sdk": "^0.7.1",
43
44
  "@safe-global/safe-apps-provider": "^0.18.0",
44
45
  "@safe-global/safe-apps-sdk": "^8.1.0",
45
46
  "@walletconnect/ethereum-provider": "^2.10.1",
@@ -48,18 +49,13 @@
48
49
  "devDependencies": {
49
50
  "lokijs": "^1.5.12",
50
51
  "msw": "^1.3.1",
51
- "@wagmi/core": "2.0.0-alpha.5"
52
+ "@wagmi/core": "2.0.0-alpha.7"
52
53
  },
53
54
  "contributors": [
54
55
  "awkweb.eth <t@wagmi.sh>",
55
56
  "jxom.eth <j@wagmi.sh>"
56
57
  ],
57
- "funding": [
58
- {
59
- "type": "github",
60
- "url": "https://github.com/sponsors/wagmi-dev"
61
- }
62
- ],
58
+ "funding": "https://github.com/sponsors/wagmi-dev",
63
59
  "keywords": [
64
60
  "react",
65
61
  "hooks",
@@ -71,9 +67,9 @@
71
67
  "abi"
72
68
  ],
73
69
  "scripts": {
74
- "clean": "rm -rf dist tsconfig.tsbuildinfo",
75
70
  "build": "pnpm run clean && pnpm run build:esm+types",
76
71
  "build:esm+types": "tsc --project tsconfig.build.json --module es2020 --outDir ./dist/esm --declaration --declarationMap --declarationDir ./dist/types",
72
+ "clean": "rm -rf dist tsconfig.tsbuildinfo",
77
73
  "test:build": "publint --strict",
78
74
  "typecheck": "tsc --noEmit"
79
75
  }
@@ -78,7 +78,7 @@ export function coinbaseWallet(parameters: CoinbaseWalletParameters) {
78
78
  return { accounts, chainId: currentChainId }
79
79
  } catch (error) {
80
80
  if (
81
- /(user closed modal|accounts received is empty)/i.test(
81
+ /(user closed modal|accounts received is empty|user denied account)/i.test(
82
82
  (error as Error).message,
83
83
  )
84
84
  )
@@ -12,6 +12,8 @@ export {
12
12
 
13
13
  export { type LedgerParameters, ledger } from '../ledger.js'
14
14
 
15
+ export { type MetaMaskParameters, metaMask } from '../metaMask.js'
16
+
15
17
  export { type SafeParameters, safe } from '../safe.js'
16
18
 
17
19
  export {
@@ -0,0 +1,316 @@
1
+ import {
2
+ EventType,
3
+ MetaMaskSDK,
4
+ type MetaMaskSDKOptions,
5
+ SDKProvider,
6
+ } from '@metamask/sdk'
7
+ import {
8
+ ChainNotConfiguredError,
9
+ createConnector,
10
+ normalizeChainId,
11
+ } from '@wagmi/core'
12
+ import type { Evaluate, ExactPartial, Omit } from '@wagmi/core/internal'
13
+ import {
14
+ type Address,
15
+ type ProviderConnectInfo,
16
+ type ProviderRpcError,
17
+ ResourceUnavailableRpcError,
18
+ RpcError,
19
+ SwitchChainError,
20
+ UserRejectedRequestError,
21
+ type WalletPermission,
22
+ getAddress,
23
+ numberToHex,
24
+ } from 'viem'
25
+
26
+ export type MetaMaskParameters = Evaluate<
27
+ ExactPartial<
28
+ Omit<
29
+ MetaMaskSDKOptions,
30
+ | 'checkInstallationImmediately'
31
+ | 'checkInstallationOnAllCalls'
32
+ | 'defaultReadOnlyChainId'
33
+ | 'readonlyRPCMap'
34
+ >
35
+ >
36
+ >
37
+
38
+ export function metaMask(parameters: MetaMaskParameters = {}) {
39
+ type Provider = SDKProvider
40
+ type Properties = {
41
+ onConnect(connectInfo: ProviderConnectInfo): void
42
+ }
43
+ type StorageItem = { 'metaMaskSDK.disconnected': true }
44
+ type Listener = Parameters<Provider['on']>[1]
45
+
46
+ let sdk: MetaMaskSDK
47
+ let walletProvider: Provider | undefined
48
+
49
+ return createConnector<Provider, Properties, StorageItem>((config) => ({
50
+ id: 'metaMaskSDK',
51
+ name: 'MetaMask',
52
+ async setup() {
53
+ const provider = await this.getProvider()
54
+ if (provider)
55
+ provider.on('connect', this.onConnect.bind(this) as Listener)
56
+ },
57
+ async connect({ chainId, isReconnecting } = {}) {
58
+ const provider = await this.getProvider()
59
+
60
+ let accounts: readonly Address[] | null = null
61
+ if (!isReconnecting) {
62
+ accounts = await this.getAccounts().catch(() => null)
63
+ const isAuthorized = !!accounts?.length
64
+ if (isAuthorized)
65
+ // Attempt to show another prompt for selecting account if already connected
66
+ try {
67
+ const permissions = (await provider.request({
68
+ method: 'wallet_requestPermissions',
69
+ params: [{ eth_accounts: {} }],
70
+ })) as WalletPermission[]
71
+ accounts = permissions[0]?.caveats?.[0]?.value?.map(getAddress)
72
+ } catch (err) {
73
+ const error = err as RpcError
74
+ // Not all injected providers support `wallet_requestPermissions` (e.g. MetaMask iOS).
75
+ // Only bubble up error if user rejects request
76
+ if (error.code === UserRejectedRequestError.code)
77
+ throw new UserRejectedRequestError(error)
78
+ // Or prompt is already open
79
+ if (error.code === ResourceUnavailableRpcError.code) throw error
80
+ }
81
+ }
82
+
83
+ try {
84
+ if (!accounts?.length) {
85
+ const requestedAccounts = (await sdk.connect()) as string[]
86
+ accounts = requestedAccounts.map(getAddress)
87
+ }
88
+
89
+ provider.removeListener(
90
+ 'connect',
91
+ this.onConnect.bind(this) as Listener,
92
+ )
93
+ provider.on(
94
+ 'accountsChanged',
95
+ this.onAccountsChanged.bind(this) as Listener,
96
+ )
97
+ provider.on('chainChanged', this.onChainChanged as Listener)
98
+ provider.on('disconnect', this.onDisconnect.bind(this) as Listener)
99
+
100
+ // Backward compatibility with older wallet (<7.3) version that return accounts before authorization
101
+ if (!sdk.isExtensionActive() && !sdk._getConnection()?.isAuthorized()) {
102
+ function waitForAuthorized() {
103
+ return new Promise((resolve) => {
104
+ const connection = sdk._getConnection()
105
+ const connector = connection?.getConnector()
106
+ connector?.once(EventType.AUTHORIZED, () => resolve(true))
107
+ })
108
+ }
109
+ await waitForAuthorized()
110
+ }
111
+
112
+ // Switch to chain if provided
113
+ let currentChainId = await this.getChainId()
114
+ if (chainId && currentChainId !== chainId) {
115
+ const chain = await this.switchChain!({ chainId }).catch(() => ({
116
+ id: currentChainId,
117
+ }))
118
+ currentChainId = chain?.id ?? currentChainId
119
+ }
120
+
121
+ await config.storage?.removeItem('metaMaskSDK.disconnected')
122
+
123
+ return { accounts, chainId: currentChainId }
124
+ } catch (err) {
125
+ const error = err as RpcError
126
+ if (error.code === UserRejectedRequestError.code)
127
+ throw new UserRejectedRequestError(error)
128
+ if (error.code === ResourceUnavailableRpcError.code)
129
+ throw new ResourceUnavailableRpcError(error)
130
+ throw error
131
+ }
132
+ },
133
+ async disconnect() {
134
+ const provider = await this.getProvider()
135
+
136
+ provider.removeListener(
137
+ 'accountsChanged',
138
+ this.onAccountsChanged.bind(this),
139
+ )
140
+ provider.removeListener('chainChanged', this.onChainChanged)
141
+ provider.removeListener('disconnect', this.onDisconnect.bind(this))
142
+ provider.on('connect', this.onConnect.bind(this) as Listener)
143
+
144
+ // Add shim signalling connector is disconnected
145
+ await config.storage?.setItem('metaMaskSDK.disconnected', true)
146
+ },
147
+ async getAccounts() {
148
+ const provider = await this.getProvider()
149
+ const accounts = (await provider.request({
150
+ method: 'eth_accounts',
151
+ })) as string[]
152
+ return accounts.map(getAddress)
153
+ },
154
+ async getChainId() {
155
+ const provider = await this.getProvider()
156
+ const chainId =
157
+ provider.chainId ?? (await provider?.request({ method: 'eth_chainId' }))
158
+ return normalizeChainId(chainId)
159
+ },
160
+ async getProvider() {
161
+ if (!walletProvider) {
162
+ if (!sdk || !sdk?.isInitialized()) {
163
+ sdk = new MetaMaskSDK({
164
+ enableDebug: false,
165
+ dappMetadata: { name: 'wagmi' },
166
+ extensionOnly: true,
167
+ useDeeplink: true,
168
+ _source: 'wagmi',
169
+ ...parameters,
170
+ checkInstallationImmediately: false,
171
+ checkInstallationOnAllCalls: false,
172
+ })
173
+ await sdk.init()
174
+ }
175
+ walletProvider = sdk.getProvider()
176
+ }
177
+ return walletProvider
178
+ },
179
+ async isAuthorized() {
180
+ try {
181
+ const isDisconnected =
182
+ // If shim exists in storage, connector is disconnected
183
+ await config.storage?.getItem('metaMaskSDK.disconnected')
184
+ if (isDisconnected) return false
185
+
186
+ const accounts = await this.getAccounts()
187
+ return !!accounts.length
188
+ } catch {
189
+ return false
190
+ }
191
+ },
192
+ async switchChain({ chainId }) {
193
+ const provider = await this.getProvider()
194
+
195
+ const chain = config.chains.find((x) => x.id === chainId)
196
+ if (!chain) throw new SwitchChainError(new ChainNotConfiguredError())
197
+
198
+ try {
199
+ await Promise.all([
200
+ provider.request({
201
+ method: 'wallet_switchEthereumChain',
202
+ params: [{ chainId: numberToHex(chainId) }],
203
+ }),
204
+ new Promise<void>((resolve) =>
205
+ config.emitter.once('change', ({ chainId: currentChainId }) => {
206
+ if (currentChainId === chainId) resolve()
207
+ }),
208
+ ),
209
+ ])
210
+ return chain
211
+ } catch (err) {
212
+ const error = err as RpcError
213
+
214
+ // Indicates chain is not added to provider
215
+ if (
216
+ error.code === 4902 ||
217
+ // Unwrapping for MetaMask Mobile
218
+ // https://github.com/MetaMask/metamask-mobile/issues/2944#issuecomment-976988719
219
+ (error as ProviderRpcError<{ originalError?: { code: number } }>)
220
+ ?.data?.originalError?.code === 4902
221
+ ) {
222
+ try {
223
+ const { default: blockExplorer, ...blockExplorers } =
224
+ chain.blockExplorers ?? {}
225
+ let blockExplorerUrls: string[] = []
226
+ if (blockExplorer)
227
+ blockExplorerUrls = [
228
+ blockExplorer.url,
229
+ ...Object.values(blockExplorers).map((x) => x.url),
230
+ ]
231
+
232
+ await provider.request({
233
+ method: 'wallet_addEthereumChain',
234
+ params: [
235
+ {
236
+ chainId: numberToHex(chainId),
237
+ chainName: chain.name,
238
+ nativeCurrency: chain.nativeCurrency,
239
+ rpcUrls: [chain.rpcUrls.public?.http[0] ?? ''],
240
+ blockExplorerUrls,
241
+ },
242
+ ],
243
+ })
244
+
245
+ const currentChainId = await this.getChainId()
246
+ if (currentChainId !== chainId)
247
+ throw new UserRejectedRequestError(
248
+ new Error('User rejected switch after adding network.'),
249
+ )
250
+
251
+ return chain
252
+ } catch (error) {
253
+ throw new UserRejectedRequestError(error as Error)
254
+ }
255
+ }
256
+
257
+ if (error.code === UserRejectedRequestError.code)
258
+ throw new UserRejectedRequestError(error)
259
+ throw new SwitchChainError(error)
260
+ }
261
+ },
262
+ async onAccountsChanged(accounts) {
263
+ // Disconnect if there are no accounts
264
+ if (accounts.length === 0) this.onDisconnect()
265
+ // Connect if emitter is listening for connect event (e.g. is disconnected and connects through wallet interface)
266
+ else if (config.emitter.listenerCount('connect')) {
267
+ const chainId = (await this.getChainId()).toString()
268
+ this.onConnect({ chainId })
269
+ await config.storage?.removeItem('metaMaskSDK.disconnected')
270
+ }
271
+ // Regular change event
272
+ else config.emitter.emit('change', { accounts: accounts.map(getAddress) })
273
+ },
274
+ onChainChanged(chain) {
275
+ const chainId = normalizeChainId(chain)
276
+ config.emitter.emit('change', { chainId })
277
+ },
278
+ async onConnect(connectInfo) {
279
+ const accounts = await this.getAccounts()
280
+ if (accounts.length === 0) return
281
+
282
+ const chainId = normalizeChainId(connectInfo.chainId)
283
+ config.emitter.emit('connect', { accounts, chainId })
284
+
285
+ const provider = await this.getProvider()
286
+ if (provider) {
287
+ provider.removeListener('connect', this.onConnect.bind(this))
288
+ provider.on('accountsChanged', this.onAccountsChanged.bind(this) as any)
289
+ provider.on('chainChanged', this.onChainChanged as any)
290
+ provider.on('disconnect', this.onDisconnect.bind(this) as any)
291
+ }
292
+ },
293
+ async onDisconnect(error) {
294
+ const provider = await this.getProvider()
295
+
296
+ // If MetaMask emits a `code: 1013` error, wait for reconnection before disconnecting
297
+ // https://github.com/MetaMask/providers/pull/120
298
+ if (error && (error as RpcError<1013>).code === 1013) {
299
+ if (provider && !!(await this.getAccounts()).length) return
300
+ }
301
+
302
+ // No need to remove 'metaMaskSDK.disconnected' from storage because `onDisconnect` is typically
303
+ // only called when the wallet is disconnected through the wallet's interface, meaning the wallet
304
+ // actually disconnected and we don't need to simulate it.
305
+ config.emitter.emit('disconnect')
306
+
307
+ provider.removeListener(
308
+ 'accountsChanged',
309
+ this.onAccountsChanged.bind(this),
310
+ )
311
+ provider.removeListener('chainChanged', this.onChainChanged)
312
+ provider.removeListener('disconnect', this.onDisconnect.bind(this))
313
+ provider.on('connect', this.onConnect.bind(this) as any)
314
+ },
315
+ }))
316
+ }
package/src/version.ts CHANGED
@@ -1 +1 @@
1
- export const version = '4.0.0-alpha.5'
1
+ export const version = '4.0.0-alpha.7'