From 7f43d594b01061d32aae5aff0c6d5a967eafa03f Mon Sep 17 00:00:00 2001 From: Dhruv Kelawala Date: Sun, 12 Nov 2023 13:00:04 +0300 Subject: [PATCH 1/6] fix: throw account () error Fixed in starknet-react: https://github.com/apibara/starknet-react/blob/d992ae7b98ce9b60805221ef3c93fb783fafcf0c/packages/core/src/hooks/useAccount.ts#L96 --- src/connectors/argentMobile/index.ts | 12 ++---------- src/connectors/injected/index.ts | 8 ++------ src/connectors/webwallet/index.ts | 12 ++---------- 3 files changed, 6 insertions(+), 26 deletions(-) diff --git a/src/connectors/argentMobile/index.ts b/src/connectors/argentMobile/index.ts index efb672d..a84f2d7 100644 --- a/src/connectors/argentMobile/index.ts +++ b/src/connectors/argentMobile/index.ts @@ -103,17 +103,9 @@ export class ArgentMobileConnector extends Connector { this._wallet = null } - async account(): Promise { - /* - Don't throw an exception if the wallet is not connected. - This is needed because when argentMobile and webwallet connectors are used together with starknet-react, - it would always try to retrieve the account since the connectors are always available (and throw an exception since the value is null) - - https://github.com/apibara/starknet-react/blob/226e4cb1d8e9b478dc57d45a98a59a57733572bb/packages/core/src/hooks/useAccount.ts#L92 - - */ + async account(): Promise { if (!this._wallet || !this._wallet.account) { - return null + throw new ConnectorNotConnectedError() } return this._wallet.account as AccountInterface diff --git a/src/connectors/injected/index.ts b/src/connectors/injected/index.ts index 183e997..818f5ce 100644 --- a/src/connectors/injected/index.ts +++ b/src/connectors/injected/index.ts @@ -160,17 +160,13 @@ export class InjectedConnector extends Connector { } } - async account(): Promise { + async account(): Promise { await this.ensureWallet() - if (!this._wallet) { + if (!this._wallet || !this._wallet.account) { throw new ConnectorNotConnectedError() } - if (!this._wallet.account) { - return null - } - return this._wallet.account } diff --git a/src/connectors/webwallet/index.ts b/src/connectors/webwallet/index.ts index bd30153..47c3aa6 100644 --- a/src/connectors/webwallet/index.ts +++ b/src/connectors/webwallet/index.ts @@ -120,19 +120,11 @@ export class WebWalletConnector extends Connector { this._wallet = _wallet } - /* - Don't throw an exception if the wallet is not connected. - This is needed because when argentMobile and webwallet connectors are used together with starknet-react, - it would always try to retrieve the account since the connectors are always available (and throw an exception since the value is null) - - https://github.com/apibara/starknet-react/blob/226e4cb1d8e9b478dc57d45a98a59a57733572bb/packages/core/src/hooks/useAccount.ts#L92 - - */ - async account(): Promise { + async account(): Promise { this._wallet = _wallet if (!this._wallet || !this._wallet.account) { - return null + throw new ConnectorNotConnectedError() } return this._wallet.account as unknown as AccountInterface From 0770d3f4cd60d9648dccd2519536a8de07bfee85 Mon Sep 17 00:00:00 2001 From: Dhruv Kelawala Date: Sun, 12 Nov 2023 13:39:08 +0300 Subject: [PATCH 2/6] fix: minor errors --- src/connectors/argentMobile/modal/adapter.ts | 2 +- src/connectors/argentMobile/modal/index.ts | 3 ++- src/connectors/argentMobile/modal/login.ts | 10 +++++++++- src/connectors/webwallet/helpers/trpc.ts | 2 +- .../webwallet/starknetWindowObject/account.ts | 6 +++++- src/main.ts | 14 ++++++++++---- src/modal/Modal.svelte | 2 +- tsconfig.json | 6 ++++-- 8 files changed, 33 insertions(+), 12 deletions(-) diff --git a/src/connectors/argentMobile/modal/adapter.ts b/src/connectors/argentMobile/modal/adapter.ts index 637f46c..c0be7c3 100644 --- a/src/connectors/argentMobile/modal/adapter.ts +++ b/src/connectors/argentMobile/modal/adapter.ts @@ -63,7 +63,7 @@ export abstract class NamespaceAdapter { }: SessionTypes.Struct) => { const chain = this.formatChainId(this.chainId) if (requiredNamespaces) { - return !!requiredNamespaces[this.namespace]?.chains.includes(chain) + return !!requiredNamespaces[this.namespace]?.chains?.includes(chain) } return !!namespaces?.[this.namespace]?.accounts.some((account) => account.startsWith(chain), diff --git a/src/connectors/argentMobile/modal/index.ts b/src/connectors/argentMobile/modal/index.ts index 35231f9..1c799a5 100644 --- a/src/connectors/argentMobile/modal/index.ts +++ b/src/connectors/argentMobile/modal/index.ts @@ -11,4 +11,5 @@ export type { StarknetWindowObject, IArgentLoginOptions } export const getStarknetWindowObject = async ( options: IArgentLoginOptions, -): Promise => login(options, StarknetAdapter) +): Promise => + login(options, StarknetAdapter) diff --git a/src/connectors/argentMobile/modal/login.ts b/src/connectors/argentMobile/modal/login.ts index 46e1a1d..8db7d64 100644 --- a/src/connectors/argentMobile/modal/login.ts +++ b/src/connectors/argentMobile/modal/login.ts @@ -37,7 +37,14 @@ export const login = async ( walletConnect, }: IArgentLoginOptions, Adapter: new (options: NamespaceAdapterOptions) => TAdapter, -): Promise => { +): Promise => { + if (!bridgeUrl) { + throw new Error("bridgeUrl is required") + } + + if (!mobileUrl) { + throw new Error("mobileUrl is required") + } argentModal.bridgeUrl = bridgeUrl argentModal.mobileUrl = mobileUrl argentModal.type = modalType @@ -101,6 +108,7 @@ export const login = async ( } catch (error) { console.error("@argent/login::error") argentModal.closeModal() + return null } } diff --git a/src/connectors/webwallet/helpers/trpc.ts b/src/connectors/webwallet/helpers/trpc.ts index ce50a01..b1b5f72 100644 --- a/src/connectors/webwallet/helpers/trpc.ts +++ b/src/connectors/webwallet/helpers/trpc.ts @@ -43,7 +43,7 @@ export const setPopupOptions = ({ : parentTop + parentHeight / 2 - height / 2 popupOrigin = origin ?? popupOrigin - popupLocation = location ?? location + popupLocation = location ?? popupLocation popupParams = `width=${width},height=${height},top=${y},left=${x},toolbar=no,menubar=no,scrollbars=no,location=no,status=no,popup=1` } diff --git a/src/connectors/webwallet/starknetWindowObject/account.ts b/src/connectors/webwallet/starknetWindowObject/account.ts index 2fde840..5a3a525 100644 --- a/src/connectors/webwallet/starknetWindowObject/account.ts +++ b/src/connectors/webwallet/starknetWindowObject/account.ts @@ -61,7 +61,11 @@ export class MessageAccount extends Account implements AccountInterface { height: EXECUTE_POPUP_HEIGHT, location: "/review", }) - if (calls[0] && calls[0].entrypoint === "use_offchain_session") { + if ( + Array.isArray(calls) && + calls[0] && + calls[0].entrypoint === "use_offchain_session" + ) { setPopupOptions({ width: 1, height: 1, diff --git a/src/main.ts b/src/main.ts index dea60ac..ddbb659 100644 --- a/src/main.ts +++ b/src/main.ts @@ -68,7 +68,9 @@ export const connect = async ({ if (wallet) { const connector = availableConnectors.find((c) => c.id === lastWalletId) await connector?.connect() - selectedConnector = connector + if (connector) { + selectedConnector = connector + } return wallet } // otherwise fallback to modal } @@ -93,11 +95,13 @@ export const connect = async ({ dappName, callback: async (value: StarknetWindowObject | null) => { try { - if (value.id !== "argentWebWallet") { + if (value !== null && value.id !== "argentWebWallet") { setStarknetLastConnectedWallet(value.id) } selectedConnector = - availableConnectors.find((c) => c.id === value.id) ?? null + availableConnectors.find( + (c) => value !== null && c.id === value.id, + ) ?? null resolve(value) } finally { setTimeout(() => modal.$destroy()) @@ -116,7 +120,9 @@ export const getSelectedConnectorWallet = () => export const disconnect = async (options: DisconnectOptions = {}) => { removeStarknetLastConnectedWallet() - await selectedConnector.disconnect() + if (selectedConnector) { + await selectedConnector.disconnect() + } selectedConnector = null return sn.disconnect(options) diff --git a/src/modal/Modal.svelte b/src/modal/Modal.svelte index 5cc346c..b84de23 100644 --- a/src/modal/Modal.svelte +++ b/src/modal/Modal.svelte @@ -51,7 +51,7 @@ darkModeControlClass = "" } - if (isInAppBrowser) { + if (isInAppBrowser && window?.starknet_argentX) { try { const enabledValue = await sn.enable(window?.starknet_argentX) callback(enabledValue ?? window?.starknet_argentX) diff --git a/tsconfig.json b/tsconfig.json index 4a1139e..b458ac4 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -16,7 +16,9 @@ */ "allowJs": true, "checkJs": true, - "declaration": true + "declaration": true, + "verbatimModuleSyntax": false }, - "include": ["src"] + "include": ["src"], + "exclude": ["node_modules"] } From 536ee1ab1420cd43c884953e7b561d425976f1bf Mon Sep 17 00:00:00 2001 From: Dhruv Kelawala Date: Sun, 12 Nov 2023 15:46:53 +0300 Subject: [PATCH 3/6] fix: icons --- src/connectors/injected/constants.ts | 6 ++---- src/connectors/webwallet/index.ts | 6 +++--- src/helpers/mapModalWallets.ts | 19 +++++++++++++------ src/modal/ConnectorButton.svelte | 10 ++++++---- src/modal/Modal.svelte | 2 +- src/types/modal.ts | 4 ++-- 6 files changed, 27 insertions(+), 20 deletions(-) diff --git a/src/connectors/injected/constants.ts b/src/connectors/injected/constants.ts index 296d2cb..36d0e95 100644 --- a/src/connectors/injected/constants.ts +++ b/src/connectors/injected/constants.ts @@ -2,7 +2,5 @@ export const ARGENT_X_ICON = ` // 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. -export const WALLET_NOT_FOUND_ICON_LIGHT = - "" -export const WALLET_NOT_FOUND_ICON_DARK = - "" +export const WALLET_NOT_FOUND_ICON_LIGHT = `` +export const WALLET_NOT_FOUND_ICON_DARK = `` diff --git a/src/connectors/webwallet/index.ts b/src/connectors/webwallet/index.ts index 47c3aa6..f5d5d02 100644 --- a/src/connectors/webwallet/index.ts +++ b/src/connectors/webwallet/index.ts @@ -16,7 +16,7 @@ import { UserNotConnectedError, UserRejectedRequestError, } from "../../errors" -import { DEFAULT_WEBWALLET_URL } from "./constants" +import { DEFAULT_WEBWALLET_ICON, DEFAULT_WEBWALLET_URL } from "./constants" import { getWebWalletStarknetObject } from "./starknetWindowObject/getWebWalletStarknetObject" let _wallet: StarknetWindowObject | null = null @@ -60,8 +60,8 @@ export class WebWalletConnector extends Connector { get icon(): ConnectorIcons { return { - light: DEFAULT_WEBWALLET_URL, - dark: DEFAULT_WEBWALLET_URL, + light: DEFAULT_WEBWALLET_ICON, + dark: DEFAULT_WEBWALLET_ICON, } } diff --git a/src/helpers/mapModalWallets.ts b/src/helpers/mapModalWallets.ts index 9aeb4c0..5b53336 100644 --- a/src/helpers/mapModalWallets.ts +++ b/src/helpers/mapModalWallets.ts @@ -2,6 +2,7 @@ import type { StarknetWindowObject, WalletProvider } from "get-starknet-core" import { Connector } from "../connectors/connector" import { ARGENT_X_ICON } from "../connectors/injected/constants" import type { ModalWallet, StoreVersion } from "../types/modal" +import { isString } from "lodash-es" interface SetConnectorsExpandedParams { availableConnectors: Connector[] @@ -25,13 +26,15 @@ export const mapModalWallets = ({ } return availableConnectors - .map((c) => { + .map((c) => { const installed = installedWallets.find((w) => w.id === c.id) if (installed) { + const installedIcon = + installed.id === "argentX" ? ARGENT_X_ICON : installed.icon return { name: installed.name, id: installed.id, - icon: installed.id === "argentX" ? ARGENT_X_ICON : installed.icon, + icon: { light: installedIcon, dark: installedIcon }, connector: c, } } @@ -44,10 +47,13 @@ export const mapModalWallets = ({ if (discovery) { const { downloads } = discovery + + const discoveryIcon = + discovery.id === "argentX" ? ARGENT_X_ICON : discovery.icon return { name: discovery.name, id: discovery.id, - icon: discovery.id === "argentX" ? ARGENT_X_ICON : discovery.icon, + icon: { light: discoveryIcon, dark: discoveryIcon }, connector: c, download: downloads[storeVersion as keyof typeof downloads], } @@ -62,9 +68,10 @@ export const mapModalWallets = ({ id: c.id, icon: c.icon, connector: c, - title: "title" in c ? c.title : undefined, - subtitle: "subtitle" in c ? c.subtitle : undefined, + title: "title" in c && isString(c.title) ? c.title : undefined, + subtitle: + "subtitle" in c && isString(c.subtitle) ? c.subtitle : undefined, } }) - .filter((c) => c !== null) as ModalWallet[] + .filter((c): c is ModalWallet => c !== null) } diff --git a/src/modal/ConnectorButton.svelte b/src/modal/ConnectorButton.svelte index 121c9f2..81cb1f4 100644 --- a/src/modal/ConnectorButton.svelte +++ b/src/modal/ConnectorButton.svelte @@ -3,9 +3,11 @@ import type { Connector } from "../connectors/connector" export let wallet: ModalWallet + export let theme: "light" | "dark" | null = null export let cb: (value: Connector | null) => Promise = async () => {} export let loadingItem: string | false = false - const isSvg = wallet.icon.startsWith(" {#if wallet.download} @@ -36,7 +38,7 @@

Install {wallet.name}

- {wallet.name} + {wallet.name} {:else} @@ -91,9 +93,9 @@ Loading... {:else if isSvg} -
{@html wallet.icon}
+
{@html icon}
{:else} - {wallet?.name} + {wallet?.name} {/if} {/if} diff --git a/src/modal/Modal.svelte b/src/modal/Modal.svelte index b84de23..c80d88e 100644 --- a/src/modal/Modal.svelte +++ b/src/modal/Modal.svelte @@ -140,7 +140,7 @@
    {#each modalWallets as wallet} - + {/each}
diff --git a/src/types/modal.ts b/src/types/modal.ts index cef77eb..91608dd 100644 --- a/src/types/modal.ts +++ b/src/types/modal.ts @@ -1,5 +1,5 @@ import type { GetWalletOptions } from "get-starknet-core" -import type { Connector } from "../connectors/connector" +import type { Connector, ConnectorIcons } from "../connectors/connector" import type { ArgentMobileConnectorOptions } from "../connectors/argentMobile" export type StoreVersion = "chrome" | "firefox" | "edge" @@ -17,7 +17,7 @@ export interface ConnectOptions extends GetWalletOptions { export type ModalWallet = { name: string id: string - icon: string + icon: ConnectorIcons download?: string subtitle?: string title?: string From 4d1a81b6f07d18d20aada7a803fa125f8e8f08b7 Mon Sep 17 00:00:00 2001 From: Dhruv Kelawala Date: Sun, 12 Nov 2023 15:56:19 +0300 Subject: [PATCH 4/6] fix: await ensureWallet --- src/connectors/argentMobile/index.ts | 2 +- src/connectors/injected/index.ts | 6 ++++-- src/connectors/webwallet/index.ts | 2 +- 3 files changed, 6 insertions(+), 4 deletions(-) diff --git a/src/connectors/argentMobile/index.ts b/src/connectors/argentMobile/index.ts index a84f2d7..abe2b03 100644 --- a/src/connectors/argentMobile/index.ts +++ b/src/connectors/argentMobile/index.ts @@ -112,7 +112,7 @@ export class ArgentMobileConnector extends Connector { } async chainId(): Promise { - this.ensureWallet() + await this.ensureWallet() if (!this._wallet) { throw new ConnectorNotConnectedError() diff --git a/src/connectors/injected/index.ts b/src/connectors/injected/index.ts index 818f5ce..5f191f0 100644 --- a/src/connectors/injected/index.ts +++ b/src/connectors/injected/index.ts @@ -36,6 +36,8 @@ export class InjectedConnector extends Connector { } available(): boolean { + // This should be awaited ideally but it would break compatibility with + // starknet-react. Do we need to make this async? Is ensureWallet needed? this.ensureWallet() return this._wallet !== undefined } @@ -50,7 +52,7 @@ export class InjectedConnector extends Connector { } async chainId(): Promise { - this.ensureWallet() + await this.ensureWallet() if (!this._wallet) { throw new ConnectorNotConnectedError() @@ -108,7 +110,7 @@ export class InjectedConnector extends Connector { } async connect(): Promise { - this.ensureWallet() + await this.ensureWallet() if (!this._wallet) { throw new ConnectorNotFoundError() diff --git a/src/connectors/webwallet/index.ts b/src/connectors/webwallet/index.ts index f5d5d02..f5514ed 100644 --- a/src/connectors/webwallet/index.ts +++ b/src/connectors/webwallet/index.ts @@ -131,7 +131,7 @@ export class WebWalletConnector extends Connector { } async chainId(): Promise { - this.ensureWallet() + await this.ensureWallet() if (!this._wallet) { throw new ConnectorNotConnectedError() From a8ca44e8dfc0cacdec737ee8e2fd93d99990d0ab Mon Sep 17 00:00:00 2001 From: bluecco Date: Mon, 13 Nov 2023 13:56:03 +0100 Subject: [PATCH 5/6] fix: argent mobile and webwallet chainId method remove ensureWallet --- src/connectors/argentMobile/index.ts | 4 +--- src/connectors/webwallet/index.ts | 4 +--- 2 files changed, 2 insertions(+), 6 deletions(-) diff --git a/src/connectors/argentMobile/index.ts b/src/connectors/argentMobile/index.ts index abe2b03..f97206c 100644 --- a/src/connectors/argentMobile/index.ts +++ b/src/connectors/argentMobile/index.ts @@ -112,9 +112,7 @@ export class ArgentMobileConnector extends Connector { } async chainId(): Promise { - await this.ensureWallet() - - if (!this._wallet) { + if (!this._wallet || !this.wallet.account || !this._wallet.provider) { throw new ConnectorNotConnectedError() } diff --git a/src/connectors/webwallet/index.ts b/src/connectors/webwallet/index.ts index f5514ed..18ddfb8 100644 --- a/src/connectors/webwallet/index.ts +++ b/src/connectors/webwallet/index.ts @@ -131,9 +131,7 @@ export class WebWalletConnector extends Connector { } async chainId(): Promise { - await this.ensureWallet() - - if (!this._wallet) { + if (!this._wallet || !this.wallet.account || !this._wallet.provider) { throw new ConnectorNotConnectedError() } From 504a6a73d5614d861d4933a69cb123a7e64be743 Mon Sep 17 00:00:00 2001 From: bluecco Date: Mon, 13 Nov 2023 14:24:35 +0100 Subject: [PATCH 6/6] chore: update ensureWallet method for injected to be compliant with starknet react --- src/connectors/injected/index.ts | 59 +++++++++++++++++++++++++++----- 1 file changed, 50 insertions(+), 9 deletions(-) diff --git a/src/connectors/injected/index.ts b/src/connectors/injected/index.ts index 5f191f0..fb29653 100644 --- a/src/connectors/injected/index.ts +++ b/src/connectors/injected/index.ts @@ -1,5 +1,4 @@ import type { StarknetWindowObject } from "get-starknet-core" -import { getStarknet } from "get-starknet-core" import { AccountInterface, constants } from "starknet" import { ConnectorNotConnectedError, @@ -43,7 +42,7 @@ export class InjectedConnector extends Connector { } async ready(): Promise { - await this.ensureWallet() + this.ensureWallet() if (!this._wallet) { return false @@ -52,7 +51,7 @@ export class InjectedConnector extends Connector { } async chainId(): Promise { - await this.ensureWallet() + this.ensureWallet() if (!this._wallet) { throw new ConnectorNotConnectedError() @@ -110,7 +109,7 @@ export class InjectedConnector extends Connector { } async connect(): Promise { - await this.ensureWallet() + this.ensureWallet() if (!this._wallet) { throw new ConnectorNotFoundError() @@ -151,7 +150,7 @@ export class InjectedConnector extends Connector { } async disconnect(): Promise { - await this.ensureWallet() + this.ensureWallet() if (!this.available()) { throw new ConnectorNotFoundError() @@ -163,7 +162,7 @@ export class InjectedConnector extends Connector { } async account(): Promise { - await this.ensureWallet() + this.ensureWallet() if (!this._wallet || !this._wallet.account) { throw new ConnectorNotConnectedError() @@ -208,12 +207,54 @@ export class InjectedConnector extends Connector { return this._wallet } - private async ensureWallet() { - const starknet = getStarknet() - const installed = await starknet.getAvailableWallets() + private ensureWallet() { + const installed = getAvailableWallets(globalThis) const wallet = installed.filter((w) => w.id === this._options.id)[0] if (wallet) { this._wallet = wallet } } } + +function getAvailableWallets(obj: Record): StarknetWindowObject[] { + 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 StarknetWindowObject + } + } + 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) { + /* empty */ + } + return false +}