Skip to main content

Documentation Index

Fetch the complete documentation index at: https://docs.rhinestone.dev/llms.txt

Use this file to discover all available pages before exploring further.

The DepositModal component handles the full deposit flow: wallet connection, source chain and token selection, amount input, and cross-chain bridging. It supports three wallet integration modes depending on how your app manages wallets.

recipient vs dappAddress

Two props look similar but serve different purposes:
  • recipient (required) — the address that receives the bridged funds on the target chain. This is a pure delivery target.
  • dappAddress (embedded wallet & QR modes) — the address that owns the deposit. The modal uses it to derive the smart account and, by default, as the delivery target. It is the user’s identity in the flow, not a destination.
In external wallet mode, the connected wallet is the owner, so only recipient is needed. In embedded and QR modes, you must pass dappAddress because the modal has no other way to know who owns the deposit.

Integration modes

External wallet

The user connects their own wallet via Reown (WalletConnect). The modal manages the wallet connection UI internally. Use this when your app doesn’t have existing wallet infrastructure. Pass a reownAppId to enable external wallet connection.
import { DepositModal } from "@rhinestone/deposit-modal";
import "@rhinestone/deposit-modal/styles.css";

<DepositModal
  isOpen={isOpen}
  onClose={() => setIsOpen(false)}
  targetChain={8453}
  targetToken="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913"
  recipient="0xYOUR_RECIPIENT_ADDRESS"
  reownAppId="YOUR_REOWN_PROJECT_ID"
  onLifecycle={(event) => event.type === "complete" && console.log(event)}
/>

Embedded wallet

Your app manages the wallet (e.g. via Privy, Dynamic, or Turnkey) and passes the owner address to the modal. The modal skips its own wallet connection UI and uses your app’s wallet context. Pass dappAddress with the wallet owner address, and onRequestConnect to trigger your app’s login flow when the modal needs a connected wallet.
import { DepositModal } from "@rhinestone/deposit-modal";
import "@rhinestone/deposit-modal/styles.css";

<DepositModal
  isOpen={isOpen}
  onClose={() => setIsOpen(false)}
  targetChain={8453}
  targetToken="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913"
  recipient="0xYOUR_RECIPIENT_ADDRESS"
  dappAddress={embeddedWalletAddress}
  onRequestConnect={() => login()}
  onLifecycle={(event) => event.type === "complete" && console.log(event)}
/>

QR code deposit

The modal displays a deposit address and QR code. The user sends funds from any wallet externally — no wallet connection is required inside the modal. Pass dappAddress without onRequestConnect or reownAppId.
import { DepositModal } from "@rhinestone/deposit-modal";
import "@rhinestone/deposit-modal/styles.css";

<DepositModal
  isOpen={isOpen}
  onClose={() => setIsOpen(false)}
  targetChain={8453}
  targetToken="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913"
  recipient="0xYOUR_RECIPIENT_ADDRESS"
  dappAddress={ownerAddress}
  onLifecycle={(event) => event.type === "complete" && console.log(event)}
/>

Transfer configuration

Control the deposit destination and optionally pre-fill source parameters.
PropTypeRequiredDescription
targetChainChain | number | "solana"YesDestination chain (viem Chain object, chain ID, or "solana")
targetTokenAddress | stringYesToken address on the destination chain (base58 mint for Solana)
recipientAddress | stringYesWhere funds are delivered on the target chain (base58 address for Solana)
defaultAmountstringNoPre-filled deposit amount. A USD number string (e.g. "25") or the sentinel "max" to fill the full available balance.
sourceChainChain | numberNoPre-selected source chain
sourceTokenAddressNoPre-selected source token
allowedRoutesRouteConfigNoRestrict available source chains and tokens
outputTokenRulesOutputTokenRule[]NoRoute the deposit to a different final token based on what the user deposited
rejectUnmappedbooleanNoReject deposits that don’t match any routing rule instead of falling back to targetToken
appBalanceUsdnumberNoThe user’s current in-app balance (USD). When set, the amount screen shows a “Balance after deposit” row (appBalanceUsd + amount).
The allowedRoutes prop accepts { sourceChains?: number[], sourceTokens?: string[] } to limit what the user can select. For supported chains and tokens, see supported chains.

Session configuration

Sessions allow the widget to execute bridging on behalf of the user’s smart account. These props control session behavior.
PropTypeDefaultDescription
sessionChainIdsnumber[]All supported chainsRestrict which chains get session keys
forceRegisterbooleanfalseForce session re-creation even if the account is already registered

Destination token routing

Deliver a different final token depending on what the user deposits. Pass outputTokenRules to map source deposits — matched by chain, token address, or symbol — to the output token delivered on the target chain. Deposits that don’t match any rule fall back to targetToken, or are rejected when rejectUnmapped is true.
<DepositModal
  // ...required props
  targetToken="0xFALLBACK_TOKEN_ADDRESS"
  outputTokenRules={[
    {
      match: { symbol: "USDC" },
      outputToken: "0x7f5c764cbc14f9669b88837ca1490cca17c31607",
    },
    {
      match: { symbol: "ETH" },
      outputToken: "0x4200000000000000000000000000000000000006",
    },
  ]}
/>
When several rules match the same deposit, the most specific one wins: chain + token outranks chain + symbol, which outranks token, then symbol, then chain alone. See token routing for the full rule semantics, priority order, and additional examples.

Post-bridge actions

Execute actions on the destination chain after bridging completes. Use this for automated swaps into vault tokens or other DeFi positions.
<DepositModal
  // ...required props
  postBridgeActions={[
    {
      type: "orderbook-swap",
      contract: "0xSWAP_CONTRACT_ADDRESS",
      outputToken: "0xVAULT_TOKEN_ADDRESS",
    },
  ]}
/>
The flow becomes: deposit on source chain → bridge to target chain → swap into the output token → deliver to the recipient.

Import from other apps

The modal can pull balances a user already holds in a supported third-party app and fund the deposit from there — no manual transfer first. Each source is off by default; opt in per source via dappImports.
<DepositModal
  // ...required props
  dappImports={{ polymarket: true }}
/>
When enabled and the connected wallet has a balance in that app, a new row (e.g. “Transfer from Polymarket”) appears on the deposit entry screen. The user picks it, and the modal pulls the funds into their smart account and bridges them to targetChain / targetToken like any other source.

Polymarket

Set dappImports={{ polymarket: true }}. The modal looks up the connected EOA’s Polymarket proxy wallet on Polygon and surfaces any pUSD or USDC.e balance. pUSD is unwrapped to USDC.e on the way out; USDC.e is what lands in the smart account and what the orchestrator bridges from. Polymarket exposes two wallet types and the modal handles both: some transfer on-chain directly from the user’s signed transaction, while others are relayed through your configured backendUrl. The default Rhinestone backend handles both with no extra setup.

Display modes

By default, the component renders as a centered modal overlay with a backdrop. Set inline={true} to render it without the overlay, fitting into your page layout.
<DepositModal
  isOpen={true}
  onClose={() => {}}
  inline={true}
  // ...other props
/>
Set closeOnOverlayClick={false} to prevent the modal from closing when the user clicks outside it.

Props reference

Required

PropTypeDescription
isOpenbooleanControls modal visibility
onClose() => voidCalled when the user closes the modal
targetChainChain | number | "solana"Destination chain (viem Chain object, chain ID, or "solana")
targetTokenAddress | stringToken address on the destination chain
recipientAddress | stringWhere funds are delivered on the target chain

Wallet

PropTypeDefaultDescription
reownAppIdstringReown project ID. Enables external wallet connection.
dappAddressAddressOwner address for embedded wallet or QR code flows
dappWalletClientWalletClientHost-provided viem wallet client for signing
dappPublicClientPublicClientHost-provided viem public client
onRequestConnect() => voidCalled when the modal needs the user to connect a wallet

Transfer

PropTypeDefaultDescription
defaultAmountstringPre-filled deposit amount. USD number string or the sentinel "max".
sourceChainChain | numberPre-selected source chain
sourceTokenAddressPre-selected source token
appBalanceUsdnumberIn-app USD balance; enables the “Balance after deposit” row
allowedRoutesRouteConfig{ sourceChains?, sourceTokens? } to restrict selection
outputTokenRulesOutputTokenRule[]Per-deposit output token routing rules
rejectUnmappedbooleanfalseReject deposits that don’t match any outputTokenRules entry
postBridgeActionsPostBridgeAction[]Actions to execute after bridging
dappImportsDappImportsConfigImport balances from third-party apps (e.g. { polymarket: true })

Session

PropTypeDefaultDescription
sessionChainIdsnumber[]All supportedChain IDs for session key creation
forceRegisterbooleanfalseForce session re-creation
signerAddressAddressDefault signerSession signer address
rhinestoneApiKeystringAPI key for account setup

Backend

PropTypeDefaultDescription
backendUrlstringRhinestone production URLURL of your deposit-widget-backend instance

Display

PropTypeDefaultDescription
inlinebooleanfalseRender without modal overlay
closeOnOverlayClickbooleantrueClose modal on backdrop click
classNamestringCSS class for the modal container
themeDepositModalThemeTheme configuration
uiConfigDepositModalUIConfigUI configuration
debugbooleanfalseEnable debug logging

Callbacks

PropTypeDescription
onReady() => voidModal initialized
onLifecycle(event: DepositLifecycleEvent) => voidLifecycle event — switch on event.type (connected, submitted, complete, failed, balance-changed, smart-account-changed)
onError(data: ErrorEventData) => voidError at any stage
onEvent(event: DepositAnalyticsEvent) => voidAnalytics event
See status tracking for lifecycle event payloads.

Solana

PropTypeDefaultDescription
enableSolanabooleantrueEnable Solana wallet support and QR flows
solanaRpcUrlstringSolana mainnetCustom Solana RPC endpoint