Skip to content

Commit 02f20c2

Browse files
authored
Merge pull request #824 from vsakos/solflare-metamask-standard-adapter
Solflare: Initialize MetaMask standard adapter
2 parents 89e864e + 8390ba1 commit 02f20c2

File tree

7 files changed

+251
-1
lines changed

7 files changed

+251
-1
lines changed

.changeset/clever-jeans-shout.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@solana/wallet-adapter-solflare': patch
3+
---
4+
5+
Add Solflare MetaMask Snap support

packages/wallets/solflare/package.json

+4-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,10 @@
3636
},
3737
"dependencies": {
3838
"@solana/wallet-adapter-base": "workspace:^",
39-
"@solflare-wallet/sdk": "^1.3.0"
39+
"@solana/wallet-standard-chains": "^1.1.0",
40+
"@solflare-wallet/metamask-sdk": "^1.0.2",
41+
"@solflare-wallet/sdk": "^1.3.0",
42+
"@wallet-standard/wallet": "^1.0.1"
4043
},
4144
"devDependencies": {
4245
"@solana/web3.js": "^1.77.3",

packages/wallets/solflare/src/adapter.ts

+2
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ import {
2222
import type { Transaction, TransactionVersion, VersionedTransaction } from '@solana/web3.js';
2323
import { type Connection, PublicKey, type TransactionSignature } from '@solana/web3.js';
2424
import type { default as Solflare } from '@solflare-wallet/sdk';
25+
import { detectAndRegisterSolflareMetaMaskWallet } from './metamask/detect.js';
2526

2627
interface SolflareWindow extends Window {
2728
solflare?: {
@@ -70,6 +71,7 @@ export class SolflareWalletAdapter extends BaseMessageSignerWalletAdapter {
7071
}
7172
return false;
7273
});
74+
scopePollingDetectionStrategy(detectAndRegisterSolflareMetaMaskWallet);
7375
}
7476
}
7577

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
import type { EthereumProvider, WindowWithEthereum } from '@solflare-wallet/metamask-sdk';
2+
import { registerWallet } from '@wallet-standard/wallet';
3+
import { SolflareMetaMaskWallet } from './wallet.js';
4+
5+
let stopPolling = false;
6+
7+
/** @internal */
8+
export function detectAndRegisterSolflareMetaMaskWallet(): boolean {
9+
// If detected, stop polling.
10+
if (stopPolling) return true;
11+
(async function () {
12+
try {
13+
// Try to detect, stop polling if detected, and register the wallet.
14+
if (await isSnapProviderDetected()) {
15+
if (!stopPolling) {
16+
stopPolling = true;
17+
registerWallet(new SolflareMetaMaskWallet());
18+
}
19+
}
20+
} catch (error) {
21+
// Stop polling on unhandled errors (this should never happen).
22+
stopPolling = true;
23+
}
24+
})();
25+
// Keep polling.
26+
return false;
27+
}
28+
29+
async function isSnapProviderDetected(): Promise<boolean> {
30+
try {
31+
const provider = (window as WindowWithEthereum).ethereum;
32+
if (!provider) return false;
33+
34+
const providerProviders = provider.providers;
35+
if (providerProviders && Array.isArray(providerProviders)) {
36+
for (const provider of providerProviders) {
37+
if (await isSnapSupported(provider)) return true;
38+
}
39+
}
40+
41+
const providerDetected = provider.detected;
42+
if (providerDetected && Array.isArray(providerDetected)) {
43+
for (const provider of providerDetected) {
44+
if (await isSnapSupported(provider)) return true;
45+
}
46+
}
47+
48+
return await isSnapSupported(provider);
49+
} catch (error) {
50+
return false;
51+
}
52+
}
53+
54+
async function isSnapSupported(provider: EthereumProvider): Promise<boolean> {
55+
try {
56+
await provider.request({ method: 'wallet_getSnaps' });
57+
return true;
58+
} catch (error) {
59+
return false;
60+
}
61+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
import type { WalletIcon } from '@wallet-standard/base';
2+
3+
/** @internal */
4+
export const icon: WalletIcon =
5+
'' as const;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,149 @@
1+
import { WalletNotConnectedError } from '@solana/wallet-adapter-base';
2+
import { SOLANA_DEVNET_CHAIN, SOLANA_MAINNET_CHAIN, SOLANA_TESTNET_CHAIN } from '@solana/wallet-standard-chains';
3+
import {
4+
SolanaSignAndSendTransaction,
5+
type SolanaSignAndSendTransactionFeature,
6+
type SolanaSignAndSendTransactionMethod,
7+
SolanaSignMessage,
8+
type SolanaSignMessageFeature,
9+
type SolanaSignMessageMethod,
10+
SolanaSignTransaction,
11+
type SolanaSignTransactionFeature,
12+
type SolanaSignTransactionMethod,
13+
} from '@solana/wallet-standard-features';
14+
import type { default as SolflareMetaMask } from '@solflare-wallet/metamask-sdk';
15+
import type { Wallet } from '@wallet-standard/base';
16+
import {
17+
StandardConnect,
18+
type StandardConnectFeature,
19+
type StandardConnectMethod,
20+
StandardDisconnect,
21+
type StandardDisconnectFeature,
22+
type StandardDisconnectMethod,
23+
StandardEvents,
24+
type StandardEventsChangeProperties,
25+
type StandardEventsFeature,
26+
type StandardEventsListeners,
27+
type StandardEventsNames,
28+
type StandardEventsOnMethod,
29+
} from '@wallet-standard/features';
30+
import { icon } from './icon.js';
31+
32+
export class SolflareMetaMaskWallet implements Wallet {
33+
readonly #listeners: { [E in StandardEventsNames]?: StandardEventsListeners[E][] } = {};
34+
readonly #version = '1.0.0' as const;
35+
readonly #name = 'MetaMask' as const;
36+
readonly #icon = icon;
37+
#solflareMetaMask: SolflareMetaMask | null = null;
38+
39+
get version() {
40+
return this.#version;
41+
}
42+
43+
get name() {
44+
return this.#name;
45+
}
46+
47+
get icon() {
48+
return this.#icon;
49+
}
50+
51+
get chains() {
52+
return [SOLANA_MAINNET_CHAIN, SOLANA_DEVNET_CHAIN, SOLANA_TESTNET_CHAIN] as const;
53+
}
54+
55+
get features(): StandardConnectFeature &
56+
StandardDisconnectFeature &
57+
StandardEventsFeature &
58+
SolanaSignAndSendTransactionFeature &
59+
SolanaSignTransactionFeature &
60+
SolanaSignMessageFeature {
61+
return {
62+
[StandardConnect]: {
63+
version: '1.0.0',
64+
connect: this.#connect,
65+
},
66+
[StandardDisconnect]: {
67+
version: '1.0.0',
68+
disconnect: this.#disconnect,
69+
},
70+
[StandardEvents]: {
71+
version: '1.0.0',
72+
on: this.#on,
73+
},
74+
[SolanaSignAndSendTransaction]: {
75+
version: '1.0.0',
76+
supportedTransactionVersions: ['legacy', 0],
77+
signAndSendTransaction: this.#signAndSendTransaction,
78+
},
79+
[SolanaSignTransaction]: {
80+
version: '1.0.0',
81+
supportedTransactionVersions: ['legacy', 0],
82+
signTransaction: this.#signTransaction,
83+
},
84+
[SolanaSignMessage]: {
85+
version: '1.0.0',
86+
signMessage: this.#signMessage,
87+
},
88+
};
89+
}
90+
91+
get accounts() {
92+
return this.#solflareMetaMask ? this.#solflareMetaMask.standardAccounts : [];
93+
}
94+
95+
#on: StandardEventsOnMethod = (event, listener) => {
96+
this.#listeners[event]?.push(listener) || (this.#listeners[event] = [listener]);
97+
return (): void => this.#off(event, listener);
98+
};
99+
100+
#emit<E extends StandardEventsNames>(event: E, ...args: Parameters<StandardEventsListeners[E]>): void {
101+
// eslint-disable-next-line prefer-spread
102+
this.#listeners[event]?.forEach((listener) => listener.apply(null, args));
103+
}
104+
105+
#off<E extends StandardEventsNames>(event: E, listener: StandardEventsListeners[E]): void {
106+
this.#listeners[event] = this.#listeners[event]?.filter((existingListener) => listener !== existingListener);
107+
}
108+
109+
#connect: StandardConnectMethod = async () => {
110+
if (!this.#solflareMetaMask) {
111+
let SolflareMetaMaskClass: typeof SolflareMetaMask;
112+
try {
113+
SolflareMetaMaskClass = (await import('@solflare-wallet/metamask-sdk')).default;
114+
} catch (error: any) {
115+
throw new Error('Unable to load Solflare MetaMask SDK');
116+
}
117+
this.#solflareMetaMask = new SolflareMetaMaskClass();
118+
this.#solflareMetaMask.on('standard_change', (properties: StandardEventsChangeProperties) =>
119+
this.#emit('change', properties)
120+
);
121+
}
122+
123+
if (!this.accounts.length) {
124+
await this.#solflareMetaMask.connect();
125+
}
126+
127+
return { accounts: this.accounts };
128+
};
129+
130+
#disconnect: StandardDisconnectMethod = async () => {
131+
if (!this.#solflareMetaMask) return;
132+
await this.#solflareMetaMask.disconnect();
133+
};
134+
135+
#signAndSendTransaction: SolanaSignAndSendTransactionMethod = async (...inputs) => {
136+
if (!this.#solflareMetaMask) throw new WalletNotConnectedError();
137+
return await this.#solflareMetaMask.standardSignAndSendTransaction(...inputs);
138+
};
139+
140+
#signTransaction: SolanaSignTransactionMethod = async (...inputs) => {
141+
if (!this.#solflareMetaMask) throw new WalletNotConnectedError();
142+
return await this.#solflareMetaMask.standardSignTransaction(...inputs);
143+
};
144+
145+
#signMessage: SolanaSignMessageMethod = async (...inputs) => {
146+
if (!this.#solflareMetaMask) throw new WalletNotConnectedError();
147+
return await this.#solflareMetaMask.standardSignMessage(...inputs);
148+
};
149+
}

pnpm-lock.yaml

+25
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)