Skip to content

Commit

Permalink
Merge pull request #824 from vsakos/solflare-metamask-standard-adapter
Browse files Browse the repository at this point in the history
Solflare: Initialize MetaMask standard adapter
  • Loading branch information
jordaaash authored Sep 8, 2023
2 parents 89e864e + 8390ba1 commit 02f20c2
Show file tree
Hide file tree
Showing 7 changed files with 251 additions and 1 deletion.
5 changes: 5 additions & 0 deletions .changeset/clever-jeans-shout.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@solana/wallet-adapter-solflare': patch
---

Add Solflare MetaMask Snap support
5 changes: 4 additions & 1 deletion packages/wallets/solflare/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,10 @@
},
"dependencies": {
"@solana/wallet-adapter-base": "workspace:^",
"@solflare-wallet/sdk": "^1.3.0"
"@solana/wallet-standard-chains": "^1.1.0",
"@solflare-wallet/metamask-sdk": "^1.0.2",
"@solflare-wallet/sdk": "^1.3.0",
"@wallet-standard/wallet": "^1.0.1"
},
"devDependencies": {
"@solana/web3.js": "^1.77.3",
Expand Down
2 changes: 2 additions & 0 deletions packages/wallets/solflare/src/adapter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import {
import type { Transaction, TransactionVersion, VersionedTransaction } from '@solana/web3.js';
import { type Connection, PublicKey, type TransactionSignature } from '@solana/web3.js';
import type { default as Solflare } from '@solflare-wallet/sdk';
import { detectAndRegisterSolflareMetaMaskWallet } from './metamask/detect.js';

interface SolflareWindow extends Window {
solflare?: {
Expand Down Expand Up @@ -70,6 +71,7 @@ export class SolflareWalletAdapter extends BaseMessageSignerWalletAdapter {
}
return false;
});
scopePollingDetectionStrategy(detectAndRegisterSolflareMetaMaskWallet);
}
}

Expand Down
61 changes: 61 additions & 0 deletions packages/wallets/solflare/src/metamask/detect.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
import type { EthereumProvider, WindowWithEthereum } from '@solflare-wallet/metamask-sdk';
import { registerWallet } from '@wallet-standard/wallet';
import { SolflareMetaMaskWallet } from './wallet.js';

let stopPolling = false;

/** @internal */
export function detectAndRegisterSolflareMetaMaskWallet(): boolean {
// If detected, stop polling.
if (stopPolling) return true;
(async function () {
try {
// Try to detect, stop polling if detected, and register the wallet.
if (await isSnapProviderDetected()) {
if (!stopPolling) {
stopPolling = true;
registerWallet(new SolflareMetaMaskWallet());
}
}
} catch (error) {
// Stop polling on unhandled errors (this should never happen).
stopPolling = true;
}
})();
// Keep polling.
return false;
}

async function isSnapProviderDetected(): Promise<boolean> {
try {
const provider = (window as WindowWithEthereum).ethereum;
if (!provider) return false;

const providerProviders = provider.providers;
if (providerProviders && Array.isArray(providerProviders)) {
for (const provider of providerProviders) {
if (await isSnapSupported(provider)) return true;
}
}

const providerDetected = provider.detected;
if (providerDetected && Array.isArray(providerDetected)) {
for (const provider of providerDetected) {
if (await isSnapSupported(provider)) return true;
}
}

return await isSnapSupported(provider);
} catch (error) {
return false;
}
}

async function isSnapSupported(provider: EthereumProvider): Promise<boolean> {
try {
await provider.request({ method: 'wallet_getSnaps' });
return true;
} catch (error) {
return false;
}
}
5 changes: 5 additions & 0 deletions packages/wallets/solflare/src/metamask/icon.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
import type { WalletIcon } from '@wallet-standard/base';

/** @internal */
export const icon: WalletIcon =
'' as const;
149 changes: 149 additions & 0 deletions packages/wallets/solflare/src/metamask/wallet.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,149 @@
import { WalletNotConnectedError } from '@solana/wallet-adapter-base';
import { SOLANA_DEVNET_CHAIN, SOLANA_MAINNET_CHAIN, SOLANA_TESTNET_CHAIN } from '@solana/wallet-standard-chains';
import {
SolanaSignAndSendTransaction,
type SolanaSignAndSendTransactionFeature,
type SolanaSignAndSendTransactionMethod,
SolanaSignMessage,
type SolanaSignMessageFeature,
type SolanaSignMessageMethod,
SolanaSignTransaction,
type SolanaSignTransactionFeature,
type SolanaSignTransactionMethod,
} from '@solana/wallet-standard-features';
import type { default as SolflareMetaMask } from '@solflare-wallet/metamask-sdk';
import type { Wallet } from '@wallet-standard/base';
import {
StandardConnect,
type StandardConnectFeature,
type StandardConnectMethod,
StandardDisconnect,
type StandardDisconnectFeature,
type StandardDisconnectMethod,
StandardEvents,
type StandardEventsChangeProperties,
type StandardEventsFeature,
type StandardEventsListeners,
type StandardEventsNames,
type StandardEventsOnMethod,
} from '@wallet-standard/features';
import { icon } from './icon.js';

export class SolflareMetaMaskWallet implements Wallet {
readonly #listeners: { [E in StandardEventsNames]?: StandardEventsListeners[E][] } = {};
readonly #version = '1.0.0' as const;
readonly #name = 'MetaMask' as const;
readonly #icon = icon;
#solflareMetaMask: SolflareMetaMask | null = null;

get version() {
return this.#version;
}

get name() {
return this.#name;
}

get icon() {
return this.#icon;
}

get chains() {
return [SOLANA_MAINNET_CHAIN, SOLANA_DEVNET_CHAIN, SOLANA_TESTNET_CHAIN] as const;
}

get features(): StandardConnectFeature &
StandardDisconnectFeature &
StandardEventsFeature &
SolanaSignAndSendTransactionFeature &
SolanaSignTransactionFeature &
SolanaSignMessageFeature {
return {
[StandardConnect]: {
version: '1.0.0',
connect: this.#connect,
},
[StandardDisconnect]: {
version: '1.0.0',
disconnect: this.#disconnect,
},
[StandardEvents]: {
version: '1.0.0',
on: this.#on,
},
[SolanaSignAndSendTransaction]: {
version: '1.0.0',
supportedTransactionVersions: ['legacy', 0],
signAndSendTransaction: this.#signAndSendTransaction,
},
[SolanaSignTransaction]: {
version: '1.0.0',
supportedTransactionVersions: ['legacy', 0],
signTransaction: this.#signTransaction,
},
[SolanaSignMessage]: {
version: '1.0.0',
signMessage: this.#signMessage,
},
};
}

get accounts() {
return this.#solflareMetaMask ? this.#solflareMetaMask.standardAccounts : [];
}

#on: StandardEventsOnMethod = (event, listener) => {
this.#listeners[event]?.push(listener) || (this.#listeners[event] = [listener]);
return (): void => this.#off(event, listener);
};

#emit<E extends StandardEventsNames>(event: E, ...args: Parameters<StandardEventsListeners[E]>): void {
// eslint-disable-next-line prefer-spread
this.#listeners[event]?.forEach((listener) => listener.apply(null, args));
}

#off<E extends StandardEventsNames>(event: E, listener: StandardEventsListeners[E]): void {
this.#listeners[event] = this.#listeners[event]?.filter((existingListener) => listener !== existingListener);
}

#connect: StandardConnectMethod = async () => {
if (!this.#solflareMetaMask) {
let SolflareMetaMaskClass: typeof SolflareMetaMask;
try {
SolflareMetaMaskClass = (await import('@solflare-wallet/metamask-sdk')).default;
} catch (error: any) {
throw new Error('Unable to load Solflare MetaMask SDK');
}
this.#solflareMetaMask = new SolflareMetaMaskClass();
this.#solflareMetaMask.on('standard_change', (properties: StandardEventsChangeProperties) =>
this.#emit('change', properties)
);
}

if (!this.accounts.length) {
await this.#solflareMetaMask.connect();
}

return { accounts: this.accounts };
};

#disconnect: StandardDisconnectMethod = async () => {
if (!this.#solflareMetaMask) return;
await this.#solflareMetaMask.disconnect();
};

#signAndSendTransaction: SolanaSignAndSendTransactionMethod = async (...inputs) => {
if (!this.#solflareMetaMask) throw new WalletNotConnectedError();
return await this.#solflareMetaMask.standardSignAndSendTransaction(...inputs);
};

#signTransaction: SolanaSignTransactionMethod = async (...inputs) => {
if (!this.#solflareMetaMask) throw new WalletNotConnectedError();
return await this.#solflareMetaMask.standardSignTransaction(...inputs);
};

#signMessage: SolanaSignMessageMethod = async (...inputs) => {
if (!this.#solflareMetaMask) throw new WalletNotConnectedError();
return await this.#solflareMetaMask.standardSignMessage(...inputs);
};
}
25 changes: 25 additions & 0 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit 02f20c2

Please sign in to comment.