From ef872699fa5ae8bbb7614fc538caaac49edca092 Mon Sep 17 00:00:00 2001 From: Manan Tank Date: Wed, 14 Feb 2024 22:35:55 +0530 Subject: [PATCH] Add smart and personal wallet switcher in CW details Modal --- .../react/hooks/connection/useAutoConnect.ts | 149 +++++++++--------- .../src/react/ui/ConnectWallet/Details.tsx | 112 ++++++------- packages/thirdweb/src/wallets/index.ts | 1 + packages/thirdweb/src/wallets/smart/index.ts | 22 ++- 4 files changed, 154 insertions(+), 130 deletions(-) diff --git a/packages/thirdweb/src/react/hooks/connection/useAutoConnect.ts b/packages/thirdweb/src/react/hooks/connection/useAutoConnect.ts index 7dc29bb537d..3469d3715cb 100644 --- a/packages/thirdweb/src/react/hooks/connection/useAutoConnect.ts +++ b/packages/thirdweb/src/react/hooks/connection/useAutoConnect.ts @@ -38,75 +38,82 @@ export function AutoConnect() { connectionManager.getStoredActiveWalletId(), ]); + // if no wallets were last connected if (!lastConnectedWalletIds) { setConnectionStatus("disconnected"); return; } - // connect the last active wallet first - const lastActiveWalletConfig = wallets.find( - (w) => w.metadata.id === lastActiveWalletId, - ); + async function handleWalletConnection(walletConfig: WalletConfig) { + // if this wallet requires a personal wallet to be connected + if (walletConfig.personalWalletConfigs) { + // get saved connection params for this wallet + const savedParams = await getSavedConnectParamsFromStorage( + walletConfig.metadata.id, + ); + + // if must be an object with `personalWalletId` property + if (!isValidWithPersonalWalletConnectionOptions(savedParams)) { + throw new Error("Invalid connection params"); + } - // connect the active wallet and set it as active - if (lastActiveWalletConfig) { - // ------------ TRY ------------ - try { - // if this wallet requires a personal wallet to be connected - if (lastActiveWalletConfig.personalWalletConfigs) { - // get saved connection params for this wallet - const savedParams = await getSavedConnectParamsFromStorage( - lastActiveWalletConfig.metadata.id, - ); - - // if must be an object with `personalWalletId` property - if (!isValidWithPersonalWalletConnectionOptions(savedParams)) { - throw new Error("Invalid connection params"); - } - - // find the personal wallet config - const personalWalletConfig = - lastActiveWalletConfig.personalWalletConfigs.find( - (w) => w.metadata.id === savedParams.personalWalletId, - ); - - if (!personalWalletConfig) { - throw new Error("Personal wallet not found"); - } - - // create and auto connect the personal wallet to get personal account - const personalWallet = personalWalletConfig.create({ - client, - dappMetadata, - }); - - const personalAccount = await personalWallet.autoConnect(); - - // create wallet - const wallet = lastActiveWalletConfig.create({ - client, - dappMetadata, - }) as WalletWithPersonalAccount; - - // auto connect the wallet using the personal account - const account = await wallet.autoConnect({ - personalAccount, - }); + // find the personal wallet config + const personalWalletConfig = walletConfig.personalWalletConfigs.find( + (w) => w.metadata.id === savedParams.personalWalletId, + ); - connect(account); + if (!personalWalletConfig) { + throw new Error("Personal wallet not found"); } - // if this wallet does not require a personal wallet to be connected - else { - const wallet = lastActiveWalletConfig.create({ - client, - dappMetadata, - }); - const account = await wallet.autoConnect(); + // create and auto connect the personal wallet to get personal account + const personalWallet = personalWalletConfig.create({ + client, + dappMetadata, + }); + + const personalAccount = await personalWallet.autoConnect(); + + // create wallet + const wallet = walletConfig.create({ + client, + dappMetadata, + }) as WalletWithPersonalAccount; + + // auto connect the wallet using the personal account + const account = await wallet.autoConnect({ + personalAccount, + }); + + return account; + } + + // if this wallet does not require a personal wallet to be connected + else { + const wallet = walletConfig.create({ + client, + dappMetadata, + }); + const account = await wallet.autoConnect(); + return account; + } + } + + // connect the last active wallet and set it as active + const activeWalletConfig = wallets.find( + (w) => w.metadata.id === lastActiveWalletId, + ); + + if (activeWalletConfig) { + try { + const account = await handleWalletConnection(activeWalletConfig); + if (account) { connect(account); + } else { + setConnectionStatus("disconnected"); } } catch (e) { - console.log("failed to auto connect last active wallet"); + console.error("Failed to auto connect last active wallet"); console.error(e); setConnectionStatus("disconnected"); } @@ -114,24 +121,18 @@ export function AutoConnect() { setConnectionStatus("disconnected"); } - const otherWalletConfigs: WalletConfig[] = []; - wallets.forEach((w) => { - if (w.metadata.id === lastActiveWalletId) { - return; - } - if (lastConnectedWalletIds.includes(w.metadata.id)) { - otherWalletConfigs.push(w); - } - }); + // then connect wallets that were last connected but were not set as active + const otherWallets = wallets.filter( + (w) => + w.metadata.id !== lastActiveWalletId && + lastConnectedWalletIds.includes(w.metadata.id), + ); - // connect other wallets - otherWalletConfigs.forEach(async (config) => { - const wallet = config.create({ - client, - dappMetadata, - }); - const account = await wallet.autoConnect(); - connectionManager.setConnectedAccount(account); + otherWallets.forEach(async (config) => { + const account = await handleWalletConnection(config); + if (account) { + connectionManager.setConnectedAccount(account); + } }); }; diff --git a/packages/thirdweb/src/react/ui/ConnectWallet/Details.tsx b/packages/thirdweb/src/react/ui/ConnectWallet/Details.tsx index a2ecd868a36..13722ed9d6a 100644 --- a/packages/thirdweb/src/react/ui/ConnectWallet/Details.tsx +++ b/packages/thirdweb/src/react/ui/ConnectWallet/Details.tsx @@ -2,6 +2,7 @@ import { ExitIcon, ChevronRightIcon, TextAlignJustifyIcon, + EnterIcon, } from "@radix-ui/react-icons"; import styled from "@emotion/styled"; import { useState, useEffect } from "react"; @@ -9,6 +10,7 @@ import { useActiveAccount, useActiveWalletChainId, useDisconnect, + useSetActiveAccount, } from "../../providers/wallet-provider.js"; import { useTWLocale } from "../../providers/locale-provider.js"; import { Modal } from "../components/Modal.js"; @@ -43,6 +45,12 @@ import type { ConnectWallet_DetailsButtonOptions, ConnectWallet_DetailsModalOptions, } from "./ConnectWalletProps.js"; +import { + smartWalletMetadata, + type Account, + type WalletWithPersonalAccount, + personalAccountToSmartAccountMap, +} from "../../../wallets/index.js"; // import { walletIds } from "../../../wallets/walletIds.js"; // TEMP @@ -89,6 +97,9 @@ export const ConnectedWalletDetails: React.FC<{ account: activeAccount, }); + const [screen, setScreen] = useState("main"); + const [isOpen, setIsOpen] = useState(false); + // const activeWalletConfig = undefined; // const activeWalletConfig = useWalletConfig(); // const ensQuery = useENS(); @@ -103,19 +114,14 @@ export const ConnectedWalletDetails: React.FC<{ // string | undefined // >(undefined); - const [screen, setScreen] = useState("main"); - const [isOpen, setIsOpen] = useState(false); - // const sdk = useSDK(); - // const personalWallet = activeWallet?.getPersonalWallet() as - // | WalletInstance - // | undefined; + const personalAccount = (activeAccount?.wallet as WalletWithPersonalAccount) + ?.personalAccount; - // const personalWalletConfig = - // personalWallet && walletContext.getWalletConfig(personalWallet); - // const wrapperWalletConfig = - // wrapperWallet && walletContext.getWalletConfig(wrapperWallet); + const smartAccount = activeAccount + ? personalAccountToSmartAccountMap.get(activeAccount) + : undefined; const disableSwitchChain = !activeAccount?.wallet.switchChain; @@ -171,7 +177,11 @@ export const ConnectedWalletDetails: React.FC<{ // const walletIconUrl = // overrideWalletIconUrl || activeWalletConfig?.meta.iconURL || ""; // const avatarOrWalletIconUrl = avatarUrl || walletIconUrl; - const avatarOrWalletIconUrl = activeAccount?.wallet.metadata.iconUrl || ""; + let avatarOrWalletIconUrl = activeAccount?.wallet.metadata.iconUrl || ""; + + if (activeAccount && "isSmartWallet" in activeAccount.wallet) { + avatarOrWalletIconUrl = smartWalletMetadata.iconUrl; + } const trigger = props.detailsButton?.render ? (
@@ -399,27 +409,19 @@ export const ConnectedWalletDetails: React.FC<{ > {networkSwitcherButton} - {/* Switch to Personal Wallet for Safe */} - {/* {personalWallet && - personalWalletConfig && - !props.hideSwitchToPersonalWallet && ( - - )} */} + )} - {/* Switch to Wrapper Wallet */} - {/* {wrapperWalletConfig && wrapperWallet && ( - - )} */} + {/* Switch to Smart Wallet */} + {smartAccount && ( + + )} {/* Switch Account for Metamask */} {/* {isActuallyMetaMask && @@ -710,33 +712,33 @@ export const StyledChevronRightIcon = /* @__PURE__ */ styled( }; }); -// function WalletSwitcher({ -// wallet, -// name, -// }: { -// wallet: WalletInstance; -// name: string; -// }) { -// const walletContext = useWalletContext(); -// const locale = useTWLocale().connectWallet; +function AccountSwitcher({ + account, + name, +}: { + account: Account; + name: string; +}) { + const setActiveAccount = useSetActiveAccount(); + const locale = useTWLocale().connectWallet; -// return ( -// { -// walletContext.setConnectedWallet(wallet); -// }} -// style={{ -// fontSize: fontSize.sm, -// }} -// > -// -// -// {locale.switchTo} {name} -// -// -// ); -// } + return ( + { + setActiveAccount(account); + }} + style={{ + fontSize: fontSize.sm, + }} + > + + + {locale.switchTo} {name} + + + ); +} // const ActiveDot = /* @__PURE__ */ StyledDiv(() => { // const theme = useCustomTheme(); diff --git a/packages/thirdweb/src/wallets/index.ts b/packages/thirdweb/src/wallets/index.ts index df62c2a04d2..8b788e1d964 100644 --- a/packages/thirdweb/src/wallets/index.ts +++ b/packages/thirdweb/src/wallets/index.ts @@ -69,6 +69,7 @@ export { smartWallet, SmartWallet, smartWalletMetadata, + personalAccountToSmartAccountMap, } from "./smart/index.js"; export type { SmartWalletOptions } from "./smart/types.js"; diff --git a/packages/thirdweb/src/wallets/smart/index.ts b/packages/thirdweb/src/wallets/smart/index.ts index abfc1c136bf..e3a540d2e7d 100644 --- a/packages/thirdweb/src/wallets/smart/index.ts +++ b/packages/thirdweb/src/wallets/smart/index.ts @@ -43,6 +43,12 @@ export const smartWalletMetadata = { "", }; +/** + * We can get the personal account for given smart account but not the other way around - this map gives us the reverse lookup + * @internal + */ +export const personalAccountToSmartAccountMap = new WeakMap(); + /** * */ @@ -51,6 +57,7 @@ export class SmartWallet implements WalletWithPersonalAccount { personalAccount: Account | undefined; metadata: Wallet["metadata"]; chainId?: bigint | undefined; + isSmartWallet: true; /** * Create an instance of the SmartWallet. @@ -63,6 +70,7 @@ export class SmartWallet implements WalletWithPersonalAccount { constructor(options: SmartWalletOptions) { this.options = options; this.metadata = options.metadata || smartWalletMetadata; + this.isSmartWallet = true; } /** @@ -98,7 +106,19 @@ export class SmartWallet implements WalletWithPersonalAccount { saveConnectParamsToStorage(this.metadata.id, paramsToSave); - return smartAccount(this, { ...this.options, ...connectionOptions }); + // TODO: listen for chainChanged event on the personal wallet and emit the disconnect event on the smart wallet + + const account = await smartAccount(this, { + ...this.options, + ...connectionOptions, + }); + + personalAccountToSmartAccountMap.set( + connectionOptions.personalAccount, + account, + ); + + return account; } /**