React
Web3Modal SDK has support for Wagmi and Ethers, which will help you interact with wallets and smart contracts. Choose one of these Ethereum libraries to get started.
Installation
- Wagmi
- Wagmi v1
- Ethers
- Ethers v5
- npm
- Yarn
- Bun
- pnpm
npm install @web3modal/wagmi wagmi viem @tanstack/react-query
yarn add @web3modal/wagmi wagmi viem @tanstack/react-query
bun add @web3modal/wagmi wagmi viem @tanstack/react-query
pnpm add @web3modal/wagmi wagmi viem @tanstack/react-query
Wagmi v1 has been deprecated. Please upgrade to Wagmi v2. Read the migration guide.
- npm
- Yarn
- Bun
- pnpm
npm install @web3modal/wagmi@3.5.7 wagmi@1.4.13 viem@1.21.4
yarn add @web3modal/wagmi@3.5.7 wagmi@1.4.13 viem@1.21.4
bun add @web3modal/wagmi@3.5.7 wagmi@1.4.13 viem@1.21.4
pnpm add @web3modal/wagmi@3.5.7 wagmi@1.4.13 viem@1.21.4
- npm
- Yarn
- Bun
- pnpm
npm install @web3modal/ethers5 ethers@5.7.2
yarn add @web3modal/ethers5 ethers@5.7.2
bun add @web3modal/ethers5 ethers@5.7.2
pnpm add @web3modal/ethers5 ethers@5.7.2
- npm
- Yarn
- Bun
- pnpm
npm install @web3modal/ethers ethers
yarn add @web3modal/ethers ethers
bun add @web3modal/ethers ethers
pnpm add @web3modal/ethers ethers
Don't have a project ID?
Head over to WalletConnect Cloud and create a new project now!
Implementation
- Wagmi
- Wagmi v1
- Ethers
- Ethers v5
Web3Modal can be configured in two different ways:
- Default: For a quick integration you can use
defaultWagmiConfig
function which wraps Wagmi'screatConfig
function with predefined configuration. This includes WalletConnect, Coinbase and Injected connectors, and our Blockchain API as a transport (if the chain is not support by the Blockchain API it will fallback to the chain's default RPC).
As shown in the code example, this configuration (with the exception of the connectors) can be overridden.
- Custom: This configuration setup is great for projects that already have Wagmi integrated or that would like to have more control over the Wagmi configuration. Once Wagmi is integrated in your project you will need to add
showQrModal: false
to the WalletConnect connector and callcreateWeb3Modal
with its required parameters.
Select your preferred configuration mode below:
- Default
- Custom
On top of your app set up the following configuration, making sure that all functions are called outside any React component to avoid unwanted rerenders.
import { createWeb3Modal } from '@web3modal/wagmi/react'
import { defaultWagmiConfig } from '@web3modal/wagmi/react/config'
import { WagmiProvider } from 'wagmi'
import { arbitrum, mainnet } from 'wagmi/chains'
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
// 0. Setup queryClient
const queryClient = new QueryClient()
// 1. Get projectId at https://cloud.walletconnect.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Create wagmiConfig
const metadata = {
name: 'Web3Modal',
description: 'Web3Modal Example',
url: 'https://web3modal.com', // origin must match your domain & subdomain
icons: ['https://avatars.githubusercontent.com/u/37784886']
}
const chains = [mainnet, arbitrum] as const
const config = defaultWagmiConfig({
chains,
projectId,
metadata,
...wagmiOptions // Optional - Override createConfig parameters
})
// 3. Create modal
createWeb3Modal({
wagmiConfig: config,
projectId,
enableAnalytics: true, // Optional - defaults to your Cloud configuration
enableOnramp: true // Optional - false as default
})
export function Web3ModalProvider({ children }) {
return (
<WagmiProvider config={config}>
<QueryClientProvider client={queryClient}>{children}</QueryClientProvider>
</WagmiProvider>
)
}
On top of your app set up the following configuration, making sure that all functions are called outside any React component to avoid unwanted rerenders.
import { createWeb3Modal } from '@web3modal/wagmi/react'
import { walletConnect } from '@web3modal/wagmi'
import { http, createConfig, WagmiProvider } from 'wagmi'
import { mainnet, sepolia } from 'wagmi/chains'
import { injected, coinbaseWallet } from 'wagmi/connectors'
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
// 0. Setup queryClient
const queryClient = new QueryClient()
// 1. Get projectId at https://cloud.walletconnect.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Create wagmiConfig
const metadata = {
name: 'Web3Modal',
description: 'Web3Modal Example',
url: 'https://web3modal.com', // origin must match your domain & subdomain
icons: ['https://avatars.githubusercontent.com/u/37784886']
}
const config = createConfig({
chains: [mainnet, sepolia],
transports: {
[mainnet.id]: http(),
[sepolia.id]: http()
},
connectors: [
walletConnect({ projectId, metadata, showQrModal: false }),
injected({ shimDisconnect: true }),
coinbaseWallet({
appName: metadata.name,
appLogoUrl: metadata.icons[0]
})
]
})
// 3. Create modal
createWeb3Modal({
wagmiConfig: config,
projectId,
enableAnalytics: true, // Optional - defaults to your Cloud configuration
enableOnramp: true // Optional - false as default
})
export function ContextProvider({ children }) {
return (
<WagmiProvider config={config}>
<QueryClientProvider client={queryClient}>{children}</QueryClientProvider>
</WagmiProvider>
)
}
Make sure that the url
from the metadata
matches your domain and subdomain. This will later be used by the Verify API to tell wallets if your application has been verified or not.
You can start Web3Modal configuration using either default or custom mode.
Default mode will implement WalletConnect, Browser Wallets (injected) and Coinbase options in addition to Wagmi's public provider and WalletConnect's provider.
- Default
- Custom
On top of your app set up the following configuration, making sure that all functions are called outside any React component to avoid unwanted rerenders.
import { createWeb3Modal, defaultWagmiConfig } from '@web3modal/wagmi/react'
import { WagmiConfig } from 'wagmi'
import { arbitrum, mainnet } from 'viem/chains'
// 1. Get projectId at https://cloud.walletconnect.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Create wagmiConfig
const metadata = {
name: 'Web3Modal',
description: 'Web3Modal Example',
url: 'https://web3modal.com',
icons: ['https://avatars.githubusercontent.com/u/37784886']
}
const chains = [mainnet, arbitrum]
const wagmiConfig = defaultWagmiConfig({
chains,
projectId,
metadata,
enableAnalytics: true // Optional - defaults to your Cloud configuration
})
// 3. Create modal
createWeb3Modal({ wagmiConfig, projectId, chains })
export default function App() {
return <WagmiConfig config={wagmiConfig}>// Rest of your app...</WagmiConfig>
}
On top of your app set up the following configuration, making sure that all functions are called outside any React component to avoid unwanted rerenders.
import { createWeb3Modal } from '@web3modal/wagmi/react'
import { walletConnectProvider, EIP6963Connector } from '@web3modal/wagmi'
import { WagmiConfig, configureChains, createConfig } from 'wagmi'
import { publicProvider } from 'wagmi/providers/public'
import { mainnet } from 'viem/chains'
import { CoinbaseWalletConnector } from 'wagmi/connectors/coinbaseWallet'
import { InjectedConnector } from 'wagmi/connectors/injected'
import { WalletConnectConnector } from 'wagmi/connectors/walletConnect'
// 1. Get projectId at https://cloud.walletconnect.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Create wagmiConfig
const { chains, publicClient } = configureChains(
[mainnet],
[walletConnectProvider({ projectId }), publicProvider()]
)
const metadata = {
name: 'Web3Modal',
description: 'Web3Modal Example',
url: 'https://web3modal.com',
icons: ['https://avatars.githubusercontent.com/u/37784886']
}
const wagmiConfig = createConfig({
autoConnect: true,
connectors: [
new WalletConnectConnector({ chains, options: { projectId, showQrModal: false, metadata } }),
new EIP6963Connector({ chains }),
new InjectedConnector({ chains, options: { shimDisconnect: true } }),
new CoinbaseWalletConnector({ chains, options: { appName: metadata.name } })
],
publicClient
})
// 3. Create modal
createWeb3Modal({
wagmiConfig,
projectId,
chains,
enableAnalytics: true // Optional - defaults to your Cloud configuration
})
export default function App() {
return <WagmiConfig config={wagmiConfig}>// Rest of your app...</WagmiConfig>
}
The provided code utilizes Wagmi, a React library for wallet and provider connections, along with the Web3Modal library to create a wallet connection modal. It features various connectors:
-
WalletConnectConnector: Enables cross-device wallet connections using the WalletConnect protocol, adhering to EIP-1193.
-
EIP6963Connector: Connects to wallets supporting the EIP-6963 standard, an Ethereum wallet extension specification.
-
InjectedConnector: Interfaces with browser wallet injections, e.g., MetaMask.
-
CoinbaseWalletConnector: Specifically for Coinbase Wallet.
Additionally, there's a publicProvider
for reading blockchain data through public nodes like Infura. The walletConnectProvider
, which operates in a manner akin to the public provider but employs the Blockchain API, is additionally accessible. The WagmiConfig
component shares wagmiConfig
with child components.
On top of your app set up the following configuration, making sure that all functions are called outside any React component to avoid unwanted rerenders.
import { createWeb3Modal, defaultConfig } from '@web3modal/ethers5/react'
// 1. Get projectId
const projectId = 'YOUR_PROJECT_ID'
// 2. Set chains
const mainnet = {
chainId: 1,
name: 'Ethereum',
currency: 'ETH',
explorerUrl: 'https://etherscan.io',
rpcUrl: 'https://cloudflare-eth.com'
}
// 3. Create a metadata object
const metadata = {
name: 'My Website',
description: 'My Website description',
url: 'https://mywebsite.com', // origin must match your domain & subdomain
icons: ['https://avatars.mywebsite.com/']
}
// 4. Create Ethers config
const ethersConfig = defaultConfig({
/*Required*/
metadata,
/*Optional*/
enableEIP6963: true, // true by default
enableInjected: true, // true by default
enableCoinbase: true, // true by default
rpcUrl: '...', // used for the Coinbase SDK
defaultChainId: 1 // used for the Coinbase SDK
})
// 5. Create a Web3Modal instance
createWeb3Modal({
ethersConfig,
chains: [mainnet],
projectId,
enableAnalytics: true // Optional - defaults to your Cloud configuration
})
export default function App() {
return <YourApp />
}
Make sure that the url
from the metadata
matches your domain and subdomain. This will later be used by the Verify API to tell wallets if your application has been verified or not.
On top of your app set up the following configuration, making sure that all functions are called outside any React component to avoid unwanted rerenders.
import { createWeb3Modal, defaultConfig } from '@web3modal/ethers/react'
// 1. Get projectId
const projectId = 'YOUR_PROJECT_ID'
// 2. Set chains
const mainnet = {
chainId: 1,
name: 'Ethereum',
currency: 'ETH',
explorerUrl: 'https://etherscan.io',
rpcUrl: 'https://cloudflare-eth.com'
}
// 3. Create a metadata object
const metadata = {
name: 'My Website',
description: 'My Website description',
url: 'https://mywebsite.com', // origin must match your domain & subdomain
icons: ['https://avatars.mywebsite.com/']
}
// 4. Create Ethers config
const ethersConfig = defaultConfig({
/*Required*/
metadata,
/*Optional*/
enableEIP6963: true, // true by default
enableInjected: true, // true by default
enableCoinbase: true, // true by default
rpcUrl: '...', // used for the Coinbase SDK
defaultChainId: 1 // used for the Coinbase SDK
})
// 5. Create a Web3Modal instance
createWeb3Modal({
ethersConfig,
chains: [mainnet],
projectId,
enableAnalytics: true // Optional - defaults to your Cloud configuration
})
export default function App() {
return <YourApp />
}
Make sure that the url
from the metadata
matches your domain and subdomain. This will later be used by the Verify API to tell wallets if your application has been verified or not.
Trigger the modal
- Wagmi
- Wagmi v1
- Ethers
- Ethers v5
To open Web3Modal you can use our web component or build your own button with Web3Modal hooks.
- Web Component
- Hooks
export default function ConnectButton() {
return <w3m-button />
}
Learn more about the Web3Modal web components here
Web components are global html elements that don't require importing.
You can trigger the modal by calling the open
function from useWeb3Modal
hook.
import { useWeb3Modal } from '@web3modal/wagmi/react'
export default function ConnectButton() {
// 4. Use modal hook
const { open } = useWeb3Modal()
return (
<>
<button onClick={() => open()}>Open Connect Modal</button>
<button onClick={() => open({ view: 'Networks' })}>Open Network Modal</button>
</>
)
}
Learn more about the Web3Modal hooks here
To open Web3Modal you can use our default web components or build your own logic using Web3Modal hooks.
- Components
- Hooks
export default function ConnectButton() {
return <w3m-button />
}
Learn more about the Web3Modal web components here
Web components are global html elements that don't require importing.
You can trigger the modal by calling the open
function from useWeb3Modal
hook.
import { useWeb3Modal } from '@web3modal/wagmi/react'
export default function ConnectButton() {
// 4. Use modal hook
const { open } = useWeb3Modal()
return (
<>
<button onClick={() => open()}>Open Connect Modal</button>
<button onClick={() => open({ view: 'Networks' })}>Open Network Modal</button>
</>
)
}
Learn more about the Web3Modal hooks here
To open Web3Modal you can use our web component or build your own button with Web3Modal hooks.
- Web Component
- Hooks
export default function ConnectButton() {
return <w3m-button />
}
Learn more about the Web3Modal web components here
Web components are global html elements that don't require importing.
You can trigger the modal by calling the open
function from useWeb3Modal
hook.
import { useWeb3Modal } from '@web3modal/ethers5/react'
export default function ConnectButton() {
// 4. Use modal hook
const { open } = useWeb3Modal()
return (
<>
<button onClick={() => open()}>Open Connect Modal</button>
<button onClick={() => open({ view: 'Networks' })}>Open Network Modal</button>
</>
)
}
Learn more about the Web3Modal hooks here
To open Web3Modal you can use our web component or build your own button with Web3Modal hooks.
- Web Component
- Hooks
export default function ConnectButton() {
return <w3m-button />
}
Learn more about the Web3Modal web components here
Web components are global html elements that don't require importing.
You can trigger the modal by calling the open
function from useWeb3Modal
hook.
import { useWeb3Modal } from '@web3modal/ethers/react'
export default function ConnectButton() {
// 4. Use modal hook
const { open } = useWeb3Modal()
return (
<>
<button onClick={() => open()}>Open Connect Modal</button>
<button onClick={() => open({ view: 'Networks' })}>Open Network Modal</button>
</>
)
}
Learn more about the Web3Modal hooks here
Smart Contract Interaction
- Wagmi
- Ethers
Wagmi hooks can help us interact with wallets and smart contracts:
import { useReadContract } from 'wagmi'
import { USDTAbi } from '../abi/USDTAbi'
const USDTAddress = '0x...'
function App() {
const result = useReadContract({
abi: USDTAbi,
address: USDTAddress,
functionName: 'totalSupply'
})
}
Read more about Wagmi hooks for smart contract interaction here.
Ethers can help us interact with wallets and smart contracts:
import { useWeb3ModalProvider, useWeb3ModalAccount } from '@web3modal/ethers/react'
import { BrowserProvider, Contract, formatUnits } from 'ethers'
const USDTAddress = '0x617f3112bf5397D0467D315cC709EF968D9ba546'
// The ERC-20 Contract ABI, which is a common contract interface
// for tokens (this is the Human-Readable ABI format)
const USDTAbi = [
'function name() view returns (string)',
'function symbol() view returns (string)',
'function balanceOf(address) view returns (uint)',
'function transfer(address to, uint amount)',
'event Transfer(address indexed from, address indexed to, uint amount)'
]
function Components() {
const { address, chainId, isConnected } = useWeb3ModalAccount()
const { walletProvider } = useWeb3ModalProvider()
async function getBalance() {
if (!isConnected) throw Error('User disconnected')
const ethersProvider = new BrowserProvider(walletProvider)
const signer = await ethersProvider.getSigner()
// The Contract object
const USDTContract = new Contract(USDTAddress, USDTAbi, signer)
const USDTBalance = await USDTContract.balanceOf(address)
console.log(formatUnits(USDTBalance, 18))
}
return <button onClick={getBalance}>Get User Balance</button>
}
Video Tutorial
Was this helpful?