Skip to content

Commit

Permalink
Add walletConnect in react, export all wallets
Browse files Browse the repository at this point in the history
  • Loading branch information
MananTank committed Feb 7, 2024
1 parent c397574 commit c98c794
Show file tree
Hide file tree
Showing 7 changed files with 242 additions and 11 deletions.
7 changes: 7 additions & 0 deletions packages/thirdweb/src/react/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -35,3 +35,10 @@ export { useBlockNumber } from "./hooks/rpc/useBlockNumber.js";

// utils
export { createContractQuery } from "./utils/createQuery.js";

// wallets
export { metamaskConfig } from "./wallets/metamask/metamaskConfig.js";
export { coinbaseConfig } from "./wallets/coinbase/coinbaseConfig.js";
export { rainbowConfig } from "./wallets/rainbow/rainbowConfig.js";
export { walletConnectConfig } from "./wallets/walletConnect/walletConnectConfig.js";
export { zerionConfig } from "./wallets/zerion/zerionConfig.js";
Original file line number Diff line number Diff line change
Expand Up @@ -125,7 +125,7 @@ export const ConnectModalContent = (props: {
const screenConfig: ScreenConfig = {
setModalVisibility,
theme: typeof theme === "string" ? theme : theme.type,
goBack: handleBack,
goBack: wallets.length > 1 ? handleBack : undefined,
size: modalConfig.modalSize,
};

Expand Down
2 changes: 2 additions & 0 deletions packages/thirdweb/src/react/wallets/defaultWallets.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,13 @@ import type { WalletConfig } from "../types/wallets.js";
import { coinbaseConfig } from "./coinbase/coinbaseConfig.js";
import { metamaskConfig } from "./metamask/metamaskConfig.js";
import { rainbowConfig } from "./rainbow/rainbowConfig.js";
import { walletConnectConfig } from "./walletConnect/walletConnectConfig.js";
import { zerionConfig } from "./zerion/zerionConfig.js";

export const defaultWallets: WalletConfig[] = [
/* @__PURE__ */ metamaskConfig(),
/* @__PURE__ */ coinbaseConfig(),
/* @__PURE__ */ walletConnectConfig(),
/* @__PURE__ */ rainbowConfig(),
/* @__PURE__ */ zerionConfig(),
];
Original file line number Diff line number Diff line change
@@ -0,0 +1,215 @@
import {
walletConnectMetadata,
walletConnect,
type WalletConnectConnectionOptions,
} from "../../../wallets/index.js";
import type { WalletConnect } from "../../../wallets/wallet-connect/index.js";
import { useTWLocale } from "../../providers/locale-provider.js";
import type { ConnectUIProps, WalletConfig } from "../../types/wallets.js";
import { Img } from "../../ui/components/Img.js";
import { QRCode } from "../../ui/components/QRCode.js";
import { Spacer } from "../../ui/components/Spacer.js";
import { Spinner } from "../../ui/components/Spinner.js";
import { Container, ModalHeader, Line } from "../../ui/components/basic.js";
import { Button } from "../../ui/components/buttons.js";
import { iconSize } from "../../ui/design-system/index.js";
import { wait } from "../../utils/wait.js";
import { useState, useRef, useEffect } from "react";
import { Text } from "../../ui/components/text.js";
import { isMobile } from "../../utils/isMobile.js";
import { HeadlessConnectUI } from "../headlessConnectUI.js";

export type WalletConnectConfigOptions = {
projectId?: string;
qrModalOptions?: WalletConnectConnectionOptions["qrModalOptions"];
recommended?: boolean;
};

/**
* Integrate MetaMask wallet connection into your app.
* @param options - Options for configuring the MetaMask wallet.
* @example
* ```tsx
* <ThirdwebProvider
* client={client}>
* wallets={[ metamaskConfig() ]}
* <App />
* </ThirdwebProvider>
* ```
* @returns WalletConfig object to be passed into `ThirdwebProvider`
*/
export const walletConnectConfig = (
options?: WalletConnectConfigOptions,
): WalletConfig => {
const config: WalletConfig = {
metadata: walletConnectMetadata,
create(createOptions) {
return walletConnect({
client: createOptions.client,
dappMetadata: createOptions.dappMetadata,
});
},
connectUI(props) {
if (isMobile()) {
return <HeadlessConnectUI {...props} />;
}

return (
<ConnectUI
{...props}
wcConfig={{
projectId: options?.projectId,
qrModalOptions: options?.qrModalOptions,
}}
/>
);
},
recommended: options?.recommended,
};

return config;
};

const ConnectUI = (
props: ConnectUIProps & {
wcConfig: {
projectId?: string;
qrModalOptions?: WalletConnectConnectionOptions["qrModalOptions"];
};
},
) => {
const locale = useTWLocale().wallets.walletConnect;
const [qrCodeUri, setQrCodeUri] = useState<string | undefined>();
const { chainId, done, createInstance, walletConfig } = props;
const [isWCModalOpen, setIsWCModalOpen] = useState(false);
const { setModalVisibility } = props.screenConfig;

const handleWCModalConnect = async () => {
try {
setQrCodeUri(undefined);
setIsWCModalOpen(true);
wait(1000).then(() => {
setModalVisibility(false);
});

// TODO: fix the type in ConnectUIProps
const wallet = createInstance() as WalletConnect;

const account = await wallet.connect({
chainId: chainId,
showQrModal: true,
qrModalOptions: props.wcConfig.qrModalOptions,
optionalChains: props.chains,
});

done(account);
setModalVisibility(true);
} catch {
setModalVisibility(true);
setIsWCModalOpen(false);
}
};

return (
<Container fullHeight animate="fadein" flex="column">
<Container p="lg">
<ModalHeader
onBack={props.screenConfig.goBack}
title={walletConfig.metadata.name}
/>
</Container>

<Spacer y="sm" />

<Container flex="column" center={"both"} px="lg" expand>
{isWCModalOpen ? (
<Container
style={{
minHeight: "300px",
}}
flex="column"
center="both"
>
<Spinner size="xl" color="accentText" />
</Container>
) : (
<WalletConnectQRScanConnect
{...props}
qrCodeUri={qrCodeUri}
setQrCodeUri={setQrCodeUri}
/>
)}
</Container>

<Container p="lg">
<Text multiline center balance>
{locale.scanInstruction}
</Text>
</Container>

<Line />
<Container py="lg" flex="row" center="x">
<Button variant="link" onClick={handleWCModalConnect}>
{"Open Official WalletConnect Modal"}
</Button>
</Container>
</Container>
);
};

function WalletConnectQRScanConnect(
props: ConnectUIProps & {
qrCodeUri?: string;
setQrCodeUri: (uri: string) => void;
wcConfig: {
projectId?: string;
qrModalOptions?: WalletConnectConnectionOptions["qrModalOptions"];
};
},
) {
const {
qrCodeUri,
walletConfig,
createInstance,
done,
chainId,
setQrCodeUri,
chains,
} = props;

const scanStarted = useRef(false);
useEffect(() => {
if (scanStarted.current) {
return;
}
scanStarted.current = true;

const wallet = createInstance() as WalletConnect;

wallet
.connect({
chainId: chainId,
onDisplayUri(uri) {
setQrCodeUri(uri);
},
optionalChains: chains,
showQrModal: false,
})
.then((account) => {
done(account);
});
}, [chainId, chains, createInstance, done, setQrCodeUri]);

return (
<QRCode
qrCodeUri={qrCodeUri}
QRIcon={
<Img
width={iconSize.xxl}
height={iconSize.xxl}
src={walletConfig.metadata.iconUrl}
/>
}
/>
);
}
7 changes: 6 additions & 1 deletion packages/thirdweb/src/wallets/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -48,4 +48,9 @@ export {
} from "./injected/wallets/zerion.js";

// wallet-connect
export { walletConnect } from "./wallet-connect/index.js";
export {
walletConnect,
walletConnectMetadata,
} from "./wallet-connect/index.js";

export type { WalletConnectConnectionOptions } from "./wallet-connect/types.js";
16 changes: 9 additions & 7 deletions packages/thirdweb/src/wallets/wallet-connect/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import {
} from "viem";
import type { Address } from "abitype";
import { normalizeChainId } from "../utils/normalizeChainId.js";
import type { DAppMetaData } from "../types.js";
import type { DAppMetaData, WalletMetadata } from "../types.js";
import {
deleteConnectParamsFromStorage,
getSavedConnectParamsFromStorage,
Expand Down Expand Up @@ -61,6 +61,13 @@ type SavedConnectParams = {
pairingTopic?: string;
};

export const walletConnectMetadata: WalletMetadata = {
name: "WalletConnect",
iconUrl:
"",
id: "walletconnect",
};

/**
* Connect to a wallet using WalletConnect protocol.
* @param options - The options for connecting wallet
Expand Down Expand Up @@ -99,12 +106,7 @@ export class WalletConnect implements Wallet {
*/
constructor(options: WalletConnectCreationOptions) {
this.options = options;
this.metadata = options?.metadata || {
name: "WalletConnect",
iconUrl:
"",
id: "walletconnect",
};
this.metadata = options?.metadata || walletConnectMetadata;
}

/**
Expand Down
4 changes: 2 additions & 2 deletions packages/thirdweb/src/wallets/wallet-connect/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import type { ThirdwebClient } from "../../client/client.js";

type EthereumProviderOptions = Parameters<(typeof EthereumProvider)["init"]>[0];

type QRCodeModalOptions = Pick<
type WalletConnectQRCodeModalOptions = Pick<
NonNullable<EthereumProviderOptions["qrModalOptions"]>,
| "themeMode"
| "themeVariables"
Expand All @@ -30,7 +30,7 @@ export type WalletConnectConnectionOptions = {
optionalChains?: (number | bigint)[];
showQrModal?: boolean;
pairingTopic?: string;
qrModalOptions?: QRCodeModalOptions;
qrModalOptions?: WalletConnectQRCodeModalOptions;
onDisplayUri?: (uri: string) => void;
onSessionRequestSent?: () => void;
};

0 comments on commit c98c794

Please sign in to comment.