React
AppKit has support for Wagmi and Ethers v6 on Ethereum, @solana/web3.js on Solana and Bitcoin. Choose one of these to get started.
Installation
If you prefer referring to a video tutorial for this, please click here.
If you are setting up your React app, please do not use npx create-react-app
, as it has been deprecated. Using it may cause dependency issues.
Instead, please use Vite to create your React app. You can set it up by running npm create vite@latest
.
- Wagmi
- Ethers
- Ethers v5
- Solana
- Bitcoin
- npm
- Yarn
- Bun
- pnpm
npm install @reown/appkit @reown/appkit-adapter-wagmi wagmi viem @tanstack/react-query
yarn add @reown/appkit @reown/appkit-adapter-wagmi wagmi viem @tanstack/react-query
bun add @reown/appkit @reown/appkit-adapter-wagmi wagmi viem @tanstack/react-query
pnpm add @reown/appkit @reown/appkit-adapter-wagmi wagmi viem @tanstack/react-query
- npm
- Yarn
- Bun
- pnpm
npm install @reown/appkit @reown/appkit-adapter-ethers5 ethers@5.7.2
yarn add @reown/appkit @reown/appkit-adapter-ethers5 ethers@5.7.2
bun add @reown/appkit @reown/appkit-adapter-ethers5 ethers@5.7.2
pnpm add @reown/appkit @reown/appkit-adapter-ethers5 ethers@5.7.2
- npm
- Yarn
- Bun
- pnpm
npm install @reown/appkit @reown/appkit-adapter-ethers ethers
yarn add @reown/appkit @reown/appkit-adapter-ethers ethers
bun add @reown/appkit @reown/appkit-adapter-ethers ethers
pnpm add @reown/appkit @reown/appkit-adapter-ethers ethers
- npm
- Yarn
- Bun
- pnpm
npm install @reown/appkit @reown/appkit-adapter-solana @solana/wallet-adapter-wallets
yarn add @reown/appkit @reown/appkit-adapter-solana @solana/wallet-adapter-wallets
bun add @reown/appkit @reown/appkit-adapter-solana @solana/wallet-adapter-wallets
pnpm add @reown/appkit @reown/appkit-adapter-solana @solana/wallet-adapter-wallets
- npm
- Yarn
- Bun
- pnpm
npm install @reown/appkit @reown/appkit-adapter-bitcoin
yarn add @reown/appkit @reown/appkit-adapter-bitcoin
bun add @reown/appkit @reown/appkit-adapter-bitcoin
pnpm add @reown/appkit @reown/appkit-adapter-bitcoin
Cloud Configuration
Create a new project on reown Cloud at https://cloud.reown.com and obtain a new project ID.
Don't have a project ID?
Head over to Reown Cloud and create a new project now!
Implementation
- Wagmi
- Ethers
- Ethers v5
- Solana
- Bitcoin
For a quick integration, you can use the createAppKit
function with a unified configuration. This automatically applies the predefined configurations for different adapters like Wagmi, Ethers, or Solana, so you no longer need to manually configure each one individually. Simply pass the common parameters such as projectId
, chains
, metadata
, etc., and the function will handle the adapter-specific configurations under the hood.
This includes WalletConnect, Coinbase and Injected connectors, and the Blockchain API as a transport
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 { createAppKit } from '@reown/appkit/react'
import { WagmiProvider } from 'wagmi'
import { arbitrum, mainnet } from '@reown/appkit/networks'
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
import { WagmiAdapter } from '@reown/appkit-adapter-wagmi'
// 0. Setup queryClient
const queryClient = new QueryClient()
// 1. Get projectId from https://cloud.reown.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Create a metadata object - optional
const metadata = {
name: 'AppKit',
description: 'AppKit Example',
url: 'https://example.com', // origin must match your domain & subdomain
icons: ['https://avatars.githubusercontent.com/u/179229932']
}
// 3. Set the networks
const networks = [mainnet, arbitrum]
// 4. Create Wagmi Adapter
const wagmiAdapter = new WagmiAdapter({
networks,
projectId,
ssr: true
})
// 5. Create modal
createAppKit({
adapters: [wagmiAdapter],
networks,
projectId,
metadata,
features: {
analytics: true // Optional - defaults to your Cloud configuration
}
})
export function AppKitProvider({ children }) {
return (
<WagmiProvider config={wagmiAdapter.wagmiConfig}>
<QueryClientProvider client={queryClient}>{children}</QueryClientProvider>
</WagmiProvider>
)
}
Importing networks
Reown AppKit use Viem networks under the hood, which provide a wide variety of networks for EVM chains. You can find all the networks supported by Viem within the @reown/appkit/networks
path.
import { createAppKit } from '@reown/appkit'
import { mainnet, arbitrum, base, scroll, polygon } from '@reown/appkit/networks'
Looking to add a custom network? Check out the custom networks section.
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 { createAppKit } from '@reown/appkit/react'
import { Ethers5Adapter } from '@reown/appkit-adapter-ethers5'
import { mainnet, arbitrum } from '@reown/appkit/networks'
// 1. Get projectId
const projectId = 'YOUR_PROJECT_ID'
// 2. Create a metadata object - optional
const metadata = {
name: 'My Website',
description: 'My Website description',
url: 'https://mywebsite.com', // origin must match your domain & subdomain
icons: ['https://avatars.mywebsite.com/']
}
// 3. Create the AppKit instance
createAppKit({
adapters: [new Ethers5Adapter()],
metadata: metadata,
networks: [mainnet, arbitrum],
projectId,
features: {
analytics: true // Optional - defaults to your Cloud configuration
}
})
export default function App() {
return <YourApp /> //make sure you have configured the <appkit-button> inside
}
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 { createAppKit } from '@reown/appkit/react'
import { EthersAdapter } from '@reown/appkit-adapter-ethers'
import { arbitrum, mainnet } from '@reown/appkit/networks'
// 1. Get projectId
const projectId = 'YOUR_PROJECT_ID'
// 2. Set the networks
const networks = [arbitrum, mainnet]
// 3. Create a metadata object - optional
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 a AppKit instance
createAppKit({
adapters: [new EthersAdapter()],
networks,
metadata,
projectId,
features: {
analytics: true // Optional - defaults to your Cloud configuration
}
})
export default function App() {
return <YourApp /> // Configure the <appkit-button> or a similar button inside
}
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 { createAppKit } from '@reown/appkit/react'
import { SolanaAdapter } from '@reown/appkit-adapter-solana/react'
import { solana, solanaTestnet, solanaDevnet } from '@reown/appkit/networks'
import { PhantomWalletAdapter, SolflareWalletAdapter } from '@solana/wallet-adapter-wallets'
// 0. Set up Solana Adapter
const solanaWeb3JsAdapter = new SolanaAdapter({
wallets: [new PhantomWalletAdapter(), new SolflareWalletAdapter()]
})
// 1. Get projectId from https://cloud.reown.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Create a metadata object - optional
const metadata = {
name: 'AppKit',
description: 'AppKit Solana Example',
url: 'https://example.com', // origin must match your domain & subdomain
icons: ['https://avatars.githubusercontent.com/u/179229932']
}
// 3. Create modal
createAppKit({
adapters: [solanaWeb3JsAdapter],
networks: [solana, solanaTestnet, solanaDevnet],
metadata: metadata,
projectId,
features: {
analytics: true // Optional - defaults to your Cloud configuration
}
})
export default function App() {
return <YourApp />
}
AppKit Bitcoin provides a set of React components and hooks to easily connect Bitcoin 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 { createAppKit } from '@reown/appkit/react'
import { BitcoinAdapter } from '@reown/appkit-adapter-bitcoin'
import { bitcoin } from '@reown/appkit/networks'
// 1. Get projectId from https://cloud.reown.com
const projectId = 'YOUR_PROJECT_ID'
// 2. Set the networks
const networks = [bitcoin]
// 3. Set up Bitcoin Adapter
const bitcoinAdapter = new BitcoinAdapter({
projectId
})
// 4. Create a metadata object - optional
const metadata = {
name: 'AppKit',
description: 'AppKit Bitcoin Example',
url: 'https://example.com', // origin must match your domain & subdomain
icons: ['https://avatars.githubusercontent.com/u/179229932']
}
// 5. Create modal
createAppKit({
adapters: [bitcoinAdapter],
networks,
metadata,
projectId,
features: {
analytics: true // Optional - defaults to your Cloud configuration,
email: false,
socials: []
}
})
export default function App() {
return <YourApp />
}
Bitcoin Provider Interface
export interface BitcoinConnector extends ChainAdapterConnector, Provider {
getAccountAddresses(): Promise<BitcoinConnector.AccountAddress[]>
signMessage(params: BitcoinConnector.SignMessageParams): Promise<string>
sendTransfer(params: BitcoinConnector.SendTransferParams): Promise<string>
signPSBT(params: BitcoinConnector.SignPSBTParams): Promise<BitcoinConnector.SignPSBTResponse>
}
Parameters
- SignMessageParams
- SignMessageParams
- SignPSBTParams
export type SignMessageParams = {
/**
* The message to be signed
*/
message: string
/**
* The address to sign the message with
*/
address: string
}
export type SendTransferParams = {
/**
* The amount to be sent in satoshis
*/
amount: string
/**
* The address to send the transfer to
*/
recipient: string
}
export type SignPSBTParams = {
/**
* The PSBT to be signed, string base64 encoded
*/
psbt: string
signInputs: {
/**
* The address whose private key to use for signing.
*/
address: string
/**
* Specifies which input to sign
*/
index: number
/**
* Specifies which part(s) of the transaction the signature commits to
*/
sighashTypes: number[]
}[]
/**
* If `true`, the PSBT will be broadcasted after signing. Default is `false`.
*/
broadcast?: boolean
}
Responses
- AccountAddress
- SignPSBTResponse
export type AccountAddress = {
/**
* Public address belonging to the account.
*/
address: string
/**
* Public key for the derivation path in hex, without 0x prefix
*/
publicKey?: string
/**
* The derivation path of the address e.g. "m/84'/0'/0'/0/0"
*/
path?: string
/**
* The purpose of the address
*/
purpose: 'payment' | 'ordinal' | 'stx'
}
export type SignPSBTResponse = {
/**
* The signed PSBT, string base64 encoded
*/
psbt: string
/**
* The `string` transaction id of the broadcasted transaction or `undefined` if not broadcasted
*/
txid?: string
}
Trigger the modal
- Wagmi
- Ethers
- Ethers v5
- Solana
- Bitcoin
To open AppKit you can use our web component or build your own button with AppKit hooks.
In this example we are going to use the <appkit-button>
component.
Web components are global html elements that don't require importing.
export default function ConnectButton() {
return <appkit-button />
}
Learn more about the AppKit web components here
To open AppKit you can use our web component or build your own button with AppKit hooks.
- Web Component
- Hooks
export default function ConnectButton() {
return <appkit-button />
}
Learn more about the AppKit 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 useAppKit
hook.
import { useAppKit } from '@reown/appkit/react'
export default function ConnectButton() {
// 4. Use modal hook
const { open } = useAppKit()
return (
<>
<button onClick={() => open()}>Open Connect Modal</button>
<button onClick={() => open({ view: 'Networks' })}>Open Network Modal</button>
</>
)
}
Learn more about the AppKit hooks here
To open AppKit you can use our web component or build your own button with AppKit hooks.
- Web Component
- Hooks
export default function ConnectButton() {
return <appkit-button />
}
Learn more about the AppKit 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 useAppKit
hook.
import { useAppKit } from '@reown/appkit/react'
export default function ConnectButton() {
// 4. Use modal hook
const { open } = useAppKit()
return (
<>
<button onClick={() => open()}>Open Connect Modal</button>
<button onClick={() => open({ view: 'Networks' })}>Open Network Modal</button>
</>
)
}
Learn more about the AppKit hooks here
To open AppKit you can use our default web components or build your own logic using AppKit hooks.
In this example we are going to use the <appkit-button>
component.
Web components are global html elements that don't require importing.
export default function ConnectButton() {
return <appkit-button />
}
To open AppKit you can use our default web components or build your own logic using AppKit hooks.
In this example we are going to use the <appkit-button>
component.
Web components are global html elements that don't require importing.
export default function ConnectButton() {
return <appkit-button />
}
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 { useAppKitProvider, useAppKitAccount } from "@reown/appkit/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, isConnected } = useAppKitAccount()
const { walletProvider } = useAppKitProvider('eip155')
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>
}
@Solana/web3.js library allows for seamless interaction with wallets and smart contracts on the Solana blockchain.
For a practical example of how it works, you can refer to our lab dApp.
import {
SystemProgram,
PublicKey,
Keypair,
Transaction,
TransactionInstruction,
LAMPORTS_PER_SOL
} from '@solana/web3.js'
import { useAppKitAccount, useAppKitProvider } from '@reown/appkit/react'
import { useAppKitConnection, type Provider } from '@reown/appkit-adapter-solana/react'
function deserializeCounterAccount(data?: Buffer): { count: number } {
if (data?.byteLength !== 8) {
throw Error('Need exactly 8 bytes to deserialize counter')
}
return {
count: Number(data[0])
}
}
const { address } = useAppKitAccount()
const { connection } = useAppKitConnection()
const { walletProvider } = useAppKitProvider<Provider>('solana')
async function onIncrementCounter() {
const PROGRAM_ID = new PublicKey('Cb5aXEgXptKqHHWLifvXu5BeAuVLjojQ5ypq6CfQj1hy')
const counterKeypair = Keypair.generate()
const counter = counterKeypair.publicKey
const balance = await connection.getBalance(walletProvider.publicKey)
if (balance < LAMPORTS_PER_SOL / 100) {
throw Error('Not enough SOL in wallet')
}
const COUNTER_ACCOUNT_SIZE = 8
const allocIx: TransactionInstruction = SystemProgram.createAccount({
fromPubkey: walletProvider.publicKey,
newAccountPubkey: counter,
lamports: await connection.getMinimumBalanceForRentExemption(COUNTER_ACCOUNT_SIZE),
space: COUNTER_ACCOUNT_SIZE,
programId: PROGRAM_ID
})
const incrementIx: TransactionInstruction = new TransactionInstruction({
programId: PROGRAM_ID,
keys: [
{
pubkey: counter,
isSigner: false,
isWritable: true
}
],
data: Buffer.from([0x0])
})
const tx = new Transaction().add(allocIx).add(incrementIx)
tx.feePayer = walletProvider.publicKey
tx.recentBlockhash = (await connection.getLatestBlockhash('confirmed')).blockhash
await walletProvider.signAndSendTransaction(tx, [counterKeypair])
const counterAccountInfo = await connection.getAccountInfo(counter, {
commitment: 'confirmed'
})
if (!counterAccountInfo) {
throw new Error('Expected counter account to have been created')
}
const counterAccount = deserializeCounterAccount(counterAccountInfo?.data)
if (counterAccount.count !== 1) {
throw new Error('Expected count to have been 1')
}
console.log(`[alloc+increment] count is: ${counterAccount.count}`);
}