Skip to content

tetherto/wdk-core-react-native

@tetherto/wdk-react-native-core

Core functionality for React Native wallets, providing wallet management, balance fetching, and more.

This library uses a unique worklet bundle to run intensive cryptographic operations on a separate thread, ensuring your app's UI remains fast and responsive.

Features

  • ⛓️ Multi-Chain Support: Manage wallets across different EVM-based networks.
  • 🧩 Extensible Architecture: Add support for new blockchains and account types via worklets.
  • 🔐 Secure Storage: Automatic encryption and secure keychain storage for sensitive data.
  • ⚛️ Modern React Hooks: A simple, powerful API for integrating wallet features into your React Native app.
  • 📦 Lightweight & Modular: Generate optimized bundles with only the chains you need.

Table of Contents

Installation

1. Install the Core Library

npm install @tetherto/wdk-react-native-core

2. Install the Worklet Bundler

You need the bundler to generate the bundle file. You can install it globally or as a dev dependency in your project.

Global Install (Recommended for easy access):

npm install -g @tetherto/wdk-worklet-bundler

Or, as a Dev Dependency:

npm install --save-dev @tetherto/wdk-worklet-bundler

Bundle Configuration

A key part of this library's architecture is the Worklet Bundle.

What is the Worklet Bundle? The bundle is a separate JavaScript file (bundle.js) that contains all the core cryptographic and blockchain logic. This code runs on a dedicated, high-performance thread, completely separate from the React Native UI thread.

Why is it necessary? Running wallet operations in a separate thread is crucial for performance. It ensures that intensive tasks like signing transactions or deriving keys do not slow down or freeze your app's user interface, resulting in a smooth and responsive experience. The bundling step also allows you to create a small, optimized bundle with only the blockchain modules you actually need.

To get the bundle, use the @tetherto/wdk-worklet-bundler CLI to generate one with only the blockchain modules you need:

# 1. Initialize configuration in your React Native project
wdk-worklet-bundler init

# 2. Edit wdk.config.js to configure your networks (see example below)

# 3. Generate the bundle
wdk-worklet-bundler generate

Example wdk.config.js:

// wdk.config.js
module.exports = {
  // Define the networks you want to support and their corresponding packages.
  networks: {
    ethereum: {
      package: '@tetherto/wdk-wallet-evm-erc-4337'
    },
    bitcoin: {
      package: '@tetherto/wdk-wallet-btc'
    }
  }
};

After running wdk-worklet-bundler generate, you will see the bundle in the directory .wdk.

For the full bundler documentation, see wdk-worklet-bundler.

Quick Start

Getting started involves three main steps:

  1. Configuration: Define your networks and generate the worklet bundle.
  2. Provider Setup: Wrap your application in WdkAppProvider and pass it your configuration.
  3. Use the Hooks: Use hooks like useWalletManager and useAddresses to manage the wallet and access its data.

➡️ For a complete example, see the Full Quick Start Guide.

Guide to Hooks

The library's functionality is exposed through a set of React hooks. They are designed to separate concerns, giving you specific tools for managing the app state, wallet lifecycle, and account interactions.

useWdkApp

  • Design Rationale: Provides a global, top-level view of the library's state. It's the single source of truth for whether the underlying engine is ready, allowing your app to react safely.
  • Standard Use Case: Displaying a loading screen while the WDK initializes.
  • Snippet:
    import { useWdkApp } from '@tetherto/wdk-react-native-core';
    
    const { isReady, error } = useWdkApp();
    if (!isReady) {
      // Show a loading or splash screen while the worklet starts
    }

useWalletManager

  • Design Rationale: Exclusively handles "heavy" lifecycle actions that affect the entire wallet (create, load, import). Separating these ensures they are used deliberately, typically during app startup or in a settings screen.
  • Standard Use Case: Checking if a wallet exists on startup, creating or loading a wallet.
  • Snippet:
    import { useWdkApp, useWalletManager } from '@tetherto/wdk-react-native-core';
    
    const { isReady } = useWdkApp();
    const { createWallet, loadWallet } = useWalletManager();
    
    useEffect(() => {
      if (isReady) {
        const setup = async () => {
          await createWallet();
        };
        setup();
      }
    }, [isReady, hasWallet, createWallet, loadWallet]);

useAddresses

  • Design Rationale: Decouples the loading and management of addresses from other account operations. This provides a focused way to get a list of addresses for the active wallet.
  • Standard Use Case: Loading the addresses for the active wallet to display them.
  • Snippet:
    import { useAddresses } from '@tetherto/wdk-react-native-core';
    
    const { addresses, loadAddresses } = useAddresses();
    
    // After a wallet is loaded (e.g. in another useEffect), load addresses.
    useEffect(() => {
      if (walletIsActive) { // Replace with your app's state logic
        loadAddresses([0, 1]); // Load addresses for account 0 and 1
      }
    }, [walletIsActive, loadAddresses]);

useAccount

  • Design Rationale: Provides the actual "actor" for a specific account (e.g., account #0 on Ethereum). This is where you find methods for doing things like signing transactions or messages.
  • Standard Use Case: Getting the account object to call signMessage when a user clicks a button.
  • Snippet:
    import { useAccount } from '@tetherto/wdk-react-native-core';
    
    const { account } = useAccount({ network: 'ethereum', accountIndex: 0 });
    
    const handleSignPress = async () => {
      const signature = await account?.signMessage('Hello WDK');
      console.log(signature);
    };

useBalance

  • Design Rationale: Isolates the logic for fetching balances. It uses TanStack Query internally to automatically handle caching, refetching, and loading states, saving you from writing boilerplate.
  • Standard Use Case: Getting the balance for a native asset (like ETH). You must define your assets first.
  • Snippet:
    import { useBalance } from '@tetherto/wdk-react-native-core';
    
    // Assume 'ethAsset' is an asset object you've defined elsewhere
    const { data } = useBalance({ 
      accountIndex: 0, 
      asset: ethAsset 
    });
    
    const balanceString = data?.balance; // e.g., '1000000000000000000'

Best Practices

See Best Practices for important patterns and recommendations for building robust apps.

Architecture

See Architecture for details on the internal design.

Security

See Security for details on security features and best practices.

Troubleshooting

See Troubleshooting for common issues and solutions.

Development

See the Contributing Guide for details on how to build, test, and contribute to this project.

License

This project is licensed under the Apache License 2.0 - see the LICENSE file for details.

About

Utilities and hooks for interacting with WDK into React Native apps.

Topics

Resources

License

Contributing

Security policy

Stars

Watchers

Forks

Packages

 
 
 

Contributors