diff --git a/change/@starknet-react-core-683ce251-dd83-4e84-a241-b4e6117f44af.json b/change/@starknet-react-core-683ce251-dd83-4e84-a241-b4e6117f44af.json new file mode 100644 index 00000000..c7fb37d6 --- /dev/null +++ b/change/@starknet-react-core-683ce251-dd83-4e84-a241-b4e6117f44af.json @@ -0,0 +1,7 @@ +{ + "type": "minor", + "comment": "core: add legacy connector adapter", + "packageName": "@starknet-react/core", + "email": "francesco@ceccon.me", + "dependentChangeType": "patch" +} diff --git a/docs/components/starknet/provider.tsx b/docs/components/starknet/provider.tsx index ce83c713..e1ca756e 100644 --- a/docs/components/starknet/provider.tsx +++ b/docs/components/starknet/provider.tsx @@ -27,6 +27,7 @@ export function StarknetProvider({ includeRecommended: "always", // Randomize the order of the connectors. order: "alphabetical", + shimLegacyConnectors: ["okxwallet"], }); return ( diff --git a/packages/core/src/connectors/discovery.ts b/packages/core/src/connectors/discovery.ts index 0b11a78e..72c889c5 100644 --- a/packages/core/src/connectors/discovery.ts +++ b/packages/core/src/connectors/discovery.ts @@ -2,7 +2,7 @@ import type { StarknetWindowObject } from "get-starknet-core"; import { useCallback, useEffect, useMemo, useState } from "react"; import type { Connector } from "./base"; -import { injected } from "./helpers"; +import { injected, legacyInjected } from "./helpers"; export type UseInjectedConnectorsProps = { /** List of recommended connectors to display. */ @@ -11,6 +11,8 @@ export type UseInjectedConnectorsProps = { includeRecommended?: "always" | "onlyIfNoConnectors"; /** How to order connectors. */ order?: "random" | "alphabetical"; + /** Shim the following legacy connectors if they are detected. */ + shimLegacyConnectors?: string[]; }; export type UseInjectedConnectorsResult = { @@ -22,14 +24,20 @@ export function useInjectedConnectors({ recommended, includeRecommended = "always", order = "alphabetical", + shimLegacyConnectors = [], }: UseInjectedConnectorsProps): UseInjectedConnectorsResult { const [injectedConnectors, setInjectedConnectors] = useState([]); const refreshConnectors = useCallback(() => { const wallets = scanObjectForWallets(window); - const connectors = wallets.map((wallet) => injected({ id: wallet.id })); + const connectors = wallets.map((wallet) => { + if (shimLegacyConnectors.includes(wallet.id)) { + return legacyInjected({ id: wallet.id }); + } + return injected({ id: wallet.id }); + }); setInjectedConnectors(connectors); - }, []); + }, [shimLegacyConnectors.includes]); useEffect(() => { refreshConnectors(); diff --git a/packages/core/src/connectors/helpers.ts b/packages/core/src/connectors/helpers.ts index a69c19eb..5e97019b 100644 --- a/packages/core/src/connectors/helpers.ts +++ b/packages/core/src/connectors/helpers.ts @@ -1,4 +1,5 @@ import { InjectedConnector } from "./injected"; +import { LegacyInjectedConnector } from "./legacy"; export function argent(): InjectedConnector { return new InjectedConnector({ @@ -25,3 +26,13 @@ export function injected({ id }: { id: string }): InjectedConnector { }, }); } + +export function legacyInjected({ + id, +}: { id: string }): LegacyInjectedConnector { + return new LegacyInjectedConnector({ + options: { + id, + }, + }); +} diff --git a/packages/core/src/connectors/index.ts b/packages/core/src/connectors/index.ts index c801f83f..d192446a 100644 --- a/packages/core/src/connectors/index.ts +++ b/packages/core/src/connectors/index.ts @@ -1,5 +1,9 @@ export { Connector, type ConnectArgs } from "./base"; export { InjectedConnector, type InjectedConnectorOptions } from "./injected"; +export { + LegacyInjectedConnector, + type LegacyInjectedConnectorOptions, +} from "./legacy"; export { type UseInjectedConnectorsProps, type UseInjectedConnectorsResult, @@ -10,4 +14,4 @@ export { type MockConnectorAccounts, type MockConnectorOptions, } from "./mock"; -export { argent, braavos, injected } from "./helpers"; +export { argent, braavos, injected, legacyInjected } from "./helpers"; diff --git a/packages/core/src/connectors/legacy.ts b/packages/core/src/connectors/legacy.ts new file mode 100644 index 00000000..fe5b6009 --- /dev/null +++ b/packages/core/src/connectors/legacy.ts @@ -0,0 +1,338 @@ +import { + Permission, + type RequestFnCall, + type RpcMessage, + type RpcTypeToMessageMap, + type WalletEventListener, +} from "@starknet-io/types-js"; +import { mainnet, sepolia } from "@starknet-react/chains"; +import type { AccountInterface, ProviderInterface } from "starknet"; +import { + ConnectorNotConnectedError, + ConnectorNotFoundError, + UserNotConnectedError, + UserRejectedRequestError, +} from "../errors"; +import { + type ConnectArgs, + Connector, + type ConnectorData, + type ConnectorIcons, +} from "./base"; + +export interface LegacyStarknetWindowObject { + id: string; + name: string; + version: string; + icon: + | string + | { + dark: string; + light: string; + }; + provider?: ProviderInterface; + account?: AccountInterface; + on: WalletEventListener; + off: WalletEventListener; + enable: (options?: { starknetVersion?: "v4" | "v5" }) => Promise; + isPreauthorized: () => Promise; + selectedAddress?: string; + chainId?: string; + isConnected: boolean; +} + +/** Injected connector options. */ +export interface LegacyInjectedConnectorOptions { + /** The wallet id. */ + id: string; + /** Wallet human readable name. */ + name?: string; + /** Wallet icons. */ + icon?: ConnectorIcons; +} + +// Icons used when the injected wallet is not found and no icon is provided. +// question-mark-circle from heroicons with color changed to black/white. +const WALLET_NOT_FOUND_ICON_LIGHT = + ""; +const WALLET_NOT_FOUND_ICON_DARK = + ""; + +// Icons used when the injected wallet is not installed +// Icons from media kits +const walletIcons = { + argentX: + "", + braavos: + "", +}; + +export class LegacyInjectedConnector extends Connector { + private _wallet?: LegacyStarknetWindowObject; + private _options: LegacyInjectedConnectorOptions; + + constructor({ options }: { options: LegacyInjectedConnectorOptions }) { + super(); + this._options = options; + } + + get id(): string { + return this._options.id; + } + + get name(): string { + return this._options.name ?? this._wallet?.name ?? this._options.id; + } + + get icon(): ConnectorIcons { + const defaultIcon = { + dark: + walletIcons[this.id as keyof typeof walletIcons] || + WALLET_NOT_FOUND_ICON_DARK, + light: + walletIcons[this.id as keyof typeof walletIcons] || + WALLET_NOT_FOUND_ICON_LIGHT, + }; + + return this._options.icon || this._wallet?.icon || defaultIcon; + } + + available(): boolean { + this.ensureWallet(); + return this._wallet !== undefined; + } + + async chainId(): Promise { + this.ensureWallet(); + + if (!this._wallet || !this._wallet.provider) { + throw new ConnectorNotConnectedError(); + } + + const chainIdHex = await this._wallet.provider.getChainId(); + const chainId = BigInt(chainIdHex); + return chainId; + } + + async ready(): Promise { + this.ensureWallet(); + + if (!this._wallet) return false; + return await this._wallet.isPreauthorized(); + } + + async connect(_args: ConnectArgs = {}): Promise { + this.ensureWallet(); + + if (!this._wallet) { + throw new ConnectorNotFoundError(); + } + + let accounts: string[]; + try { + accounts = await this._wallet.enable({ starknetVersion: "v5" }); + } catch { + // NOTE: Argent v3.0.0 swallows the `.enable` call on reject, so this won't get hit. + throw new UserRejectedRequestError(); + } + + if (!this._wallet.isConnected || !this._wallet.account || !accounts) { + // NOTE: Argent v3.0.0 swallows the `.enable` call on reject, so this won't get hit. + throw new UserRejectedRequestError(); + } + + this._wallet.on( + "accountsChanged", + async (accounts: string[] | undefined) => { + if (!accounts) return; + await this.onAccountsChanged(accounts); + }, + ); + + this._wallet.on("networkChanged", (network?: string) => { + this.onNetworkChanged(network); + }); + + await this.onAccountsChanged(accounts); + + const account = this._wallet.account.address; + const chainId = await this.chainId(); + + this.emit("connect", { account, chainId }); + + return { + account, + chainId, + }; + } + + async disconnect(): Promise { + this.ensureWallet(); + + if (!this.available()) { + throw new ConnectorNotFoundError(); + } + + if (!this._wallet?.isConnected) { + throw new UserNotConnectedError(); + } + + this.emit("disconnect"); + } + + async account(): Promise { + this.ensureWallet(); + + if (!this._wallet || !this._wallet.account) { + throw new ConnectorNotConnectedError(); + } + + return this._wallet.account; + } + + async request( + call: RequestFnCall, + ): Promise { + this.ensureWallet(); + + if (!this._wallet) { + throw new ConnectorNotConnectedError(); + } + + try { + switch (call.type) { + case "wallet_getPermissions": { + if (this._wallet) { + return [Permission.ACCOUNTS]; + } + return []; + } + case "wallet_requestAccounts": { + if (this._wallet.account) { + return [this._wallet.account.address]; + } + return []; + } + case "wallet_requestChainId": { + if (this._wallet.chainId) { + return this._wallet.chainId; + } + // @ts-ignore + return null; + } + case "wallet_addInvokeTransaction": { + if (!this._wallet) { + throw new Error("Send transaction failed"); + } + + // @ts-ignore + const calls = (call.params.calls ?? []).map( + // @ts-ignore + ({ calldata, contract_address, entry_point }) => ({ + calldata, + contractAddress: contract_address, + entrypoint: entry_point, + }), + ); + // @ts-ignore + return await this._wallet.account?.execute(calls); + } + default: { + throw new Error(`Wallet API method ${call.type} is not supported.`); + } + } + } catch { + throw new UserRejectedRequestError(); + } + } + + private ensureWallet() { + const installed = getAvailableWallets(globalThis); + const wallet = installed.filter((w) => w.id === this._options.id)[0]; + if (wallet) { + this._wallet = wallet; + } + } + + private async onAccountsChanged(accounts: string[] | string): Promise { + let account: string; + if (typeof accounts === "string") { + account = accounts; + } else { + account = accounts[0]; + } + + if (account) { + const chainId = await this.chainId(); + this.emit("change", { account, chainId }); + } else { + this.emit("disconnect"); + } + } + + private onNetworkChanged(network?: string): void { + switch (network) { + // Argent + case "SN_MAIN": + this.emit("change", { chainId: mainnet.id }); + break; + case "SN_SEPOLIA": + this.emit("change", { chainId: sepolia.id }); + break; + // Braavos + case "mainnet-alpha": + this.emit("change", { chainId: mainnet.id }); + break; + case "sepolia-alpha": + this.emit("change", { chainId: sepolia.id }); + break; + default: + this.emit("change", {}); + break; + } + } +} + +function getAvailableWallets( + // biome-ignore lint: window could contain anything + obj: Record, +): LegacyStarknetWindowObject[] { + return Object.values( + Object.getOwnPropertyNames(obj).reduce< + Record + >((wallets, key) => { + if (key.startsWith("starknet")) { + const wallet = obj[key]; + + if (isWalletObject(wallet) && !wallets[wallet.id]) { + wallets[wallet.id] = wallet as LegacyStarknetWindowObject; + } + } + return wallets; + }, {}), + ); +} + +// biome-ignore lint: wallet could be anything +function isWalletObject(wallet: any): boolean { + try { + return ( + wallet && + [ + // wallet's must have methods/members, see IStarknetWindowObject + "request", + "isConnected", + "provider", + "enable", + "isPreauthorized", + "on", + "off", + "version", + "id", + "name", + "icon", + ].every((key) => key in wallet) + ); + } catch (err) {} + return false; +}