Next.js
AppKit has support for Wagmi and Ethers v6 on Ethereum and @solana/web3.js on Solana. Choose one of these Ethereum Libraries or 'Solana' to get started.
These steps are specific to Next.js app router. For other React frameworks read the React documentation.
Installation
- Wagmi
- Ethers
- Ethers v5
- Solana
- 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
- 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
Web3Modal Solana is in canary and is not recommended for production use.
- npm
- Yarn
- Bun
- pnpm
npm install @web3modal/solana
yarn add @web3modal/solana
bun add @web3modal/solana
pnpm add @web3modal/solana
Cloud Configuration
Create a new project on WalletConnect Cloud at https://cloud.walletconnect.com and obtain a new project ID.
Don't have a project ID?
Head over to WalletConnect Cloud and create a new project now!
Implementation
- Wagmi
- Ethers
- Ethers v5
- Solana
For a quick integration you can use defaultWagmiConfig
function which wraps Wagmi's createConfig
function with a predefined configuration.
This includes WalletConnect, Coinbase and Injected connectors, and the Blockchain API as a transport
Wagmi config
Create a new file for your Wagmi configuration, since we are going to be calling this function on the client and the server it cannot live inside a file with the 'use client' directive.
For this example we will create a file called config/index.tsx
outside our app directory and set up the following configuration
import { defaultWagmiConfig } from '@web3modal/wagmi/react/config'
import { cookieStorage, createStorage } from 'wagmi'
import { mainnet, sepolia } from 'wagmi/chains'
// Get projectId from https://cloud.walletconnect.com
export const projectId = process.env.NEXT_PUBLIC_PROJECT_ID
if (!projectId) throw new Error('Project ID is not defined')
const metadata = {
name: 'Web3Modal',
description: 'Web3Modal Example',
url: 'https://web3modal.com', // origin must match your domain & subdomain
icons: ['https://avatars.githubusercontent.com/u/37784886']
}
// Create wagmiConfig
const chains = [mainnet, sepolia] as const
export const config = defaultWagmiConfig({
chains,
projectId,
metadata,
ssr: true,
storage: createStorage({
storage: cookieStorage
}),
})
- Notice that we are using here the recommended configuration from Wagmi for SSR.
- Using cookies is completely optional and by default Wagmi will use
localStorage
instead if thestorage
param is not defined. - The
ssr
flag will delay the hydration of the Wagmi's store to avoid hydration mismatch errors.
Context Provider
Let's create now a context provider that will wrap our application and initialized Web3Modal (createWeb3Modal
needs to be called inside a React Client Component file).
In this example we will create a file called context/index.tsx
outside our app directory and set up the following configuration
'use client'
import React, { ReactNode } from 'react'
import { config, projectId } from '@/config'
import { createWeb3Modal } from '@web3modal/wagmi/react'
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
import { State, WagmiProvider } from 'wagmi'
// Setup queryClient
const queryClient = new QueryClient()
if (!projectId) throw new Error('Project ID is not defined')
// Create modal
createWeb3Modal({
wagmiConfig: config,
projectId,
enableAnalytics: true, // Optional - defaults to your Cloud configuration
enableOnramp: true // Optional - false as default
})
export default function Web3ModalProvider({
children,
initialState
}: {
children: ReactNode
initialState?: State
}) {
return (
<WagmiProvider config={config} initialState={initialState}>
<QueryClientProvider client={queryClient}>{children}</QueryClientProvider>
</WagmiProvider>
)
}
Layout
Next, in our app/layout.tsx
file, we will import our Web3ModalProvider
component and call the Wagmi's function cookieToInitialState
.
The initialState
returned by cookieToInitialState
, contains the optimistic values that will populate the Wagmi's store both on the server and client.
import './globals.css'
import type { Metadata } from 'next'
import { headers } from 'next/headers'
import { cookieToInitialState } from 'wagmi'
import { config } from '@/config'
import Web3ModalProvider from '@/context'
export const metadata: Metadata = {
title: 'Create Next App',
description: 'Generated by create next app'
}
export default function RootLayout({
children
}: Readonly<{
children: React.ReactNode
}>) {
const initialState = cookieToInitialState(config, headers().get('cookie'))
return (
<html lang="en">
<body>
<Web3ModalProvider initialState={initialState}>{children}</Web3ModalProvider>
</body>
</html>
)
}
In this example we will create a new file called context/web3modal.tsx
outside our app directory and set up the following configuration
'use client'
import { createWeb3Modal, defaultConfig } from '@web3modal/ethers5/react'
// 1. Get projectId at https://cloud.walletconnect.com
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
enableOnramp: true // Optional - false as default
})
export function Web3Modal({ children }) {
return children
}
Next, in your app/layout.tsx
or app/layout.jsx
file, import the custom Web3Modal component.
import './globals.css'
import { Web3Modal } from '../context/web3modal'
export const metadata = {
title: 'Web3Modal',
description: 'Web3Modal Example'
}
export default function RootLayout({ children }) {
return (
<html lang="en">
<body>
<Web3Modal>{children}</Web3Modal>
</body>
</html>
)
}
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.
In this example we will create a new file called context/web3modal.tsx
outside our app directory and set up the following configuration
'use client'
import { createWeb3Modal, defaultConfig } from '@web3modal/ethers/react'
// 1. Get projectId from https://cloud.walletconnect.com
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
enableOnramp: true // Optional - false as default
})
export function Web3Modal({ children }) {
return children
}
Next, in your app/layout.tsx
or app/layout.jsx
file, import the custom Web3Modal component.
import './globals.css'
import { Web3Modal } from '../context/web3modal'
export const metadata = {
title: 'Web3Modal',
description: 'Web3Modal Example'
}
export default function RootLayout({ children }) {
return (
<html lang="en">
<body>
<Web3Modal>{children}</Web3Modal>
</body>
</html>
)
}
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.
AppKit Solana provides a set of React components and hooks to easily connect Solana wallets with your application.
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.
// App.tsx
import { createWeb3Modal, defaultSolanaConfig } from '@web3modal/solana/react'
import { solana, solanaTestnet, solanaDevnet } from '@web3modal/solana/chains'
// 0. Setup chains
const chains = [solana, solanaTestnet, solanaDevnet]
// 1. Get projectId from https://cloud.walletconnect.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Create solanaConfig
const metadata = {
name: 'Web3Modal',
description: 'Web3Modal Solana Example',
url: 'https://web3modal.com', // origin must match your domain & subdomain
icons: ['https://avatars.githubusercontent.com/u/37784886']
}
const solanaConfig = defaultSolanaConfig({
metadata,
chains,
projectId
})
// 3. Create modal
createWeb3Modal({
solanaConfig,
chains,
projectId
})
export default function App() {
return <YourApp />
}
Trigger the modal
- Wagmi
- Ethers
- Ethers v5
- Solana
To open Web3Modal you can use our web component or build your own button with Web3Modal hooks.
In this example we are going to use the <w3m-button>
component.
Web components are global html elements that don't require importing.
export default function ConnectButton() {
return <w3m-button />
}
Learn more about the Web3Modal web components 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
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
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/solana/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
- Solana
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>
}
Extra configuration
Next.js relies on SSR. This means some specific steps are required to make Web3Modal work properly.
- Add the following code in the
next.config.js
file
// Path: next.config.js
const nextConfig = {
webpack: config => {
config.externals.push('pino-pretty', 'lokijs', 'encoding')
return config
}
}
Was this helpful?