From 72029802e0fe03093ed277c9b305e8bc49fe8e5f Mon Sep 17 00:00:00 2001 From: HoangVD2 <102846781+HoangVD2@users.noreply.github.com> Date: Tue, 12 Mar 2024 16:21:43 +0700 Subject: [PATCH] docs: create content for Libraries Integration (#8) --- .vitepress/config.ts | 10 +- developers/blocknative-xdefi-integration.md | 27 ++ developers/cosmoskit-xdefi-integration.md | 43 +++ developers/introduction.md | 6 +- developers/libraries-integration.md | 10 + developers/rainbowkit-xdefi-integration.md | 122 +++++++ .../solana-adapter-xdefi-integration.md | 344 ++++++++++++++++++ 7 files changed, 554 insertions(+), 8 deletions(-) create mode 100644 developers/cosmoskit-xdefi-integration.md create mode 100644 developers/libraries-integration.md create mode 100644 developers/rainbowkit-xdefi-integration.md create mode 100644 developers/solana-adapter-xdefi-integration.md diff --git a/.vitepress/config.ts b/.vitepress/config.ts index ea001f0b0..7aa2fe90f 100644 --- a/.vitepress/config.ts +++ b/.vitepress/config.ts @@ -422,7 +422,7 @@ function sidebarHome() { collapsed: true, items: [ { - text: "XDEFI Wallet integration", + text: "XDEFI Wallet Integration", link: "/developers/xdefi-wallet-integration", collapsed: true, items: [ @@ -499,7 +499,7 @@ function sidebarHome() { ], }, { - text: "Blockhains Integration", + text: "Blockchains Integration", link: "/developers/blockchains-integration", collapsed: true, items: [ @@ -566,7 +566,7 @@ function sidebarHome() { ], }, { - text: "Libraries integration", + text: "Libraries Integration", link: "/developers/libraries-integration", collapsed: true, items: [ @@ -576,11 +576,11 @@ function sidebarHome() { link: "/developers/blocknative-xdefi-integration", }, { - text: "Cosmos Kit", + text: "CosmosKit", link: "/developers/cosmoskit-xdefi-integration", }, { - text: "Rainbowkit", + text: "RainbowKit", link: "/developers/rainbowkit-xdefi-integration", }, { diff --git a/developers/blocknative-xdefi-integration.md b/developers/blocknative-xdefi-integration.md index ff74bfea2..aff3d4c20 100644 --- a/developers/blocknative-xdefi-integration.md +++ b/developers/blocknative-xdefi-integration.md @@ -1 +1,28 @@ # BlockNative XDEFI Integration + +First, your app need to install the core Onboard library, the injected wallets module and optionally ethers js to support browser extension and mobile wallets: + +```bash +yarn add @web3-onboard/core @web3-onboard/injected-wallets ethers +``` + +Then initialize in your app: + +```javascript +import Onboard from "@web3-onboard/core"; +import xdefiWalletModule from "@web3-onboard/xdefi"; + +// initialize the module with options +const xdefiWalletSdk = xdefiWalletModule(); + +const onboard = Onboard({ + // ... other Onboard options + wallets: [ + xdefiWalletSdk(), + //... other wallets + ], +}); + +const connectedWallets = await onboard.connectWallet(); +console.log(connectedWallets); +``` diff --git a/developers/cosmoskit-xdefi-integration.md b/developers/cosmoskit-xdefi-integration.md new file mode 100644 index 000000000..a07d6a250 --- /dev/null +++ b/developers/cosmoskit-xdefi-integration.md @@ -0,0 +1,43 @@ +# CosmosKit XDEFI Integration + +First, your app need to install 2 packages for the XDEFI: + +- `@cosmos-kit/xdefi` +- `@cosmos-kit/xdefi-extension` + +`@cosmos-kit/xdefi` export all available xdefi wallets (currently only extension available), while if you only want to add a particular one, choose `@cosmos-kit/xdefi-extension` + +> Note: all these packages export `wallets` and it's an array of `MainWalletBase` + +Take `@cosmos-kit/xdefi` for example + +### add `@cosmos-kit/xdefi` + +```bash +yarn add @cosmos-kit/xdefi +``` + +### import the wallets + +```javascript +import { wallets as xdefi } from "@cosmos-kit/xdefi"; +``` + +### add to your provider + +```javascript +function MyCosmosApp({ Component, pageProps }: AppProps) { + return ( + + + + ); +} + +export default MyCosmosApp; +``` diff --git a/developers/introduction.md b/developers/introduction.md index ba6541860..466097a56 100644 --- a/developers/introduction.md +++ b/developers/introduction.md @@ -53,7 +53,7 @@ XDEFI Wallet is also integrated in a large panel of libraries to make it accessi You can acces the list from here: -- [BlockNative](./blocknative-xdefi-integration -- [Cosmos Kit](./cosmoskit-xdefi-integration) -- [Rainbowkit](./rainbowkit-xdefi-integration) +- [BlockNative](./blocknative-xdefi-integration) +- [CosmosKit](./cosmoskit-xdefi-integration) +- [RainbowKit](./rainbowkit-xdefi-integration) - [Solana Adapter](./solana-adapter-xdefi-integration) diff --git a/developers/libraries-integration.md b/developers/libraries-integration.md new file mode 100644 index 000000000..728ba9263 --- /dev/null +++ b/developers/libraries-integration.md @@ -0,0 +1,10 @@ +# Libraries Integration + +XDEFI Wallet is also integrated in a large panel of libraries to make it accessible to any developer's needs. + +You can access the list from here: + +- [BlockNative](./blocknative-xdefi-integration) +- [CosmosKit](./cosmoskit-xdefi-integration) +- [RainbowKit](./rainbowkit-xdefi-integration) +- [Solana Adapter](./solana-adapter-xdefi-integration) diff --git a/developers/rainbowkit-xdefi-integration.md b/developers/rainbowkit-xdefi-integration.md new file mode 100644 index 000000000..15f9fb89a --- /dev/null +++ b/developers/rainbowkit-xdefi-integration.md @@ -0,0 +1,122 @@ +# RainbowKit XDEFI Integration + +You can import individual wallets from `'@rainbow-me/rainbowkit/wallets'` along with the `connectorsForWallets` function to build your own list of wallets with their necessary connectors. This way you have full control over which wallets to display, and in which order. + +For example, you can choose to only show Rainbow along with generic fallback wallets. + +```javascript +import { connectorsForWallets } from "@rainbow-me/rainbowkit"; +import { + rainbowWallet, + walletConnectWallet, +} from "@rainbow-me/rainbowkit/wallets"; + +const connectors = connectorsForWallets( + [ + { + groupName: "Recommended", + wallets: [rainbowWallet, walletConnectWallet], + }, + ], + { + appName: "My RainbowKit App", + projectId: "YOUR_PROJECT_ID", + }, +); +``` + +You can then pass your connectors to Wagmi's `createConfig`. + +```javascript +import { connectorsForWallets } from '@rainbow-me/rainbowkit'; +import { createConfig } from 'wagmi'; + +const connectors = connectorsForWallets(/* ... */); + +const config = createConfig({ + connectors, + {/* Wagmi config */} +}); + +const queryClient = new QueryClient(); + +const App = () => ( + + + + {/* Your App */} + + + +); +``` + +### Built-in XDEFI Wallets + +First, you need to install the `@rainbow-me/rainbowkit/wallets` package and then import the dApp: + +```javascript +import { xdefiWallet } from "@rainbow-me/rainbowkit/wallets"; +``` + +### Examples + +Here are examples: Show Rainbow, MetaMask, Coinbase and XDEFI along with generic fallback wallets. + +```javascript +import { connectorsForWallets } from '@rainbow-me/rainbowkit'; +import { + rainbowWallet, + xdefiWallet + metaMaskWallet, + coinbaseWallet, + walletConnectWallet, +} from '@rainbow-me/rainbowkit/wallets'; + +const connectors = connectorsForWallets( + [ + { + groupName: 'Suggested', + wallets: [ + rainbowWallet, + xdefiWallet, + metaMaskWallet, + coinbaseWallet, + walletConnectWallet, + ], + }, + ], + { appName: 'RainbowKit App', projectId: 'YOUR_PROJECT_ID' }, +); +``` + +> Reminder: The order of the wallets array defines the order in which wallets will be displayed in the UI. + +You also can use the `groupName` key to name different wallet groups. This is useful if you want to communicate to your users which wallets you recommend, as well as other possible wallets. + +Recommend Rainbow and MetaMask, but also offer Coinbase along with generic fallback wallets. + +```javascript +import { connectorsForWallets } from "@rainbow-me/rainbowkit"; +import { + rainbowWallet, + xdefiWallet, + metaMaskWallet, + coinbaseWallet, + walletConnectWallet, +} from "@rainbow-me/rainbowkit/wallets"; + +const connectors = connectorsForWallets( + [ + { + groupName: "Recommended", + wallets: [rainbowWallet, metaMaskWallet], + }, + { + groupName: "Others", + wallets: [xdefiWallet, coinbaseWallet, walletConnectWallet], + }, + ], + { appName: "RainbowKit App", projectId: "YOUR_PROJECT_ID" }, +); +``` diff --git a/developers/solana-adapter-xdefi-integration.md b/developers/solana-adapter-xdefi-integration.md new file mode 100644 index 000000000..52acccd48 --- /dev/null +++ b/developers/solana-adapter-xdefi-integration.md @@ -0,0 +1,344 @@ +# Solana Adapter XDEFI Integration + +First, your app need to install dependencies for Solana Adapter and XDEFI. These imports include event emitters, error types, transaction-related types, and the PublicKey class. + +- `@solana/wallet-adapter-base` +- `@solana/web3.js` + +Then import dependencies and declare the `XDEFIWallet` and `XDEFIWalletWindow` interfaces: + +```typescript +import type { EventEmitter, WalletName } from "@solana/wallet-adapter-base"; +import { + BaseMessageSignerWalletAdapter, + scopePollingDetectionStrategy, + WalletAccountError, + WalletConnectionError, + WalletDisconnectedError, + WalletDisconnectionError, + WalletNotConnectedError, + WalletNotReadyError, + WalletPublicKeyError, + WalletReadyState, + WalletSignMessageError, + WalletSignTransactionError, +} from "@solana/wallet-adapter-base"; +import type { + SendOptions, + Transaction, + TransactionSignature, +} from "@solana/web3.js"; +import { PublicKey } from "@solana/web3.js"; + +interface XDEFIWalletEvents { + connect(...args: unknown[]): unknown; + disconnect(...args: unknown[]): unknown; +} + +interface XDEFIWallet extends EventEmitter { + isXDEFI?: boolean; + publicKey?: { toBytes(): Uint8Array }; + isConnected: boolean; + signTransaction(transaction: Transaction): Promise; + signAllTransactions(transactions: Transaction[]): Promise; + signAndSendTransaction( + transaction: Transaction, + options?: SendOptions, + ): Promise<{ signature: TransactionSignature }>; + signMessage(message: Uint8Array): Promise<{ signature: Uint8Array }>; + connect(): Promise; + disconnect(): Promise; +} + +interface XDEFIWalletWindow extends Window { + xfi?: { + solana?: XDEFIWallet; + }; +} +``` + +Next, constructor a new class that extends `BaseMessageSignerWalletAdapter` and implements the `XDEFIWallet` interface: + +```typescript +declare const window: XDEFIWalletWindow; + +export interface XDEFIWalletAdapterConfig {} + +export const XDEFIWalletName = "XDEFI" as WalletName<"XDEFI">; + +export class XDEFIWalletAdapter extends BaseMessageSignerWalletAdapter { + name = XDEFIWalletName; + url = "https://xdefi.io"; + icon = + ""; + readonly supportedTransactionVersions = null; + + private _connecting: boolean; + private _wallet: XDEFIWallet | null; + private _publicKey: PublicKey | null; + private _readyState: WalletReadyState = + typeof window === "undefined" || typeof document === "undefined" + ? WalletReadyState.Unsupported + : WalletReadyState.NotDetected; + + constructor(config: XDEFIWalletAdapterConfig = {}) { + super(); + this._connecting = false; + this._wallet = null; + this._publicKey = null; + + if (this._readyState !== WalletReadyState.Unsupported) { + scopePollingDetectionStrategy(() => { + if (window.xfi?.solana?.isXDEFI) { + this._readyState = WalletReadyState.Installed; + this.emit("readyStateChange", this._readyState); + return true; + } + return false; + }); + } + } + // Define the `connect`, `disconnect`, `signTransaction`, `signAllTransactions`, `signAndSendTransaction`, and `signMessage` methods like other wallet adapters. +} +``` + +Be sure to handle the `connect` and `disconnect` methods to update the `publicKey` and `isConnected` properties of the wallet adapter. + +### Example + +Here is an example of a Solana Adapter for XDEFI integration: + +```typescript +import type { EventEmitter, WalletName } from "@solana/wallet-adapter-base"; +import { + BaseMessageSignerWalletAdapter, + scopePollingDetectionStrategy, + WalletAccountError, + WalletConnectionError, + WalletDisconnectedError, + WalletDisconnectionError, + WalletNotConnectedError, + WalletNotReadyError, + WalletPublicKeyError, + WalletReadyState, + WalletSignMessageError, + WalletSignTransactionError, +} from "@solana/wallet-adapter-base"; +import type { + SendOptions, + Transaction, + TransactionSignature, +} from "@solana/web3.js"; +import { PublicKey } from "@solana/web3.js"; + +interface XDEFIWalletEvents { + connect(...args: unknown[]): unknown; + disconnect(...args: unknown[]): unknown; +} + +interface XDEFIWallet extends EventEmitter { + isXDEFI?: boolean; + publicKey?: { toBytes(): Uint8Array }; + isConnected: boolean; + signTransaction(transaction: Transaction): Promise; + signAllTransactions(transactions: Transaction[]): Promise; + signAndSendTransaction( + transaction: Transaction, + options?: SendOptions, + ): Promise<{ signature: TransactionSignature }>; + signMessage(message: Uint8Array): Promise<{ signature: Uint8Array }>; + connect(): Promise; + disconnect(): Promise; +} + +interface XDEFIWalletWindow extends Window { + xfi?: { + solana?: XDEFIWallet; + }; +} + +declare const window: XDEFIWalletWindow; + +export interface XDEFIWalletAdapterConfig {} + +export const XDEFIWalletName = "XDEFI" as WalletName<"XDEFI">; + +export class XDEFIWalletAdapter extends BaseMessageSignerWalletAdapter { + name = XDEFIWalletName; + url = "https://xdefi.io"; + icon = + ""; + readonly supportedTransactionVersions = null; + + private _connecting: boolean; + private _wallet: XDEFIWallet | null; + private _publicKey: PublicKey | null; + private _readyState: WalletReadyState = + typeof window === "undefined" || typeof document === "undefined" + ? WalletReadyState.Unsupported + : WalletReadyState.NotDetected; + + constructor(config: XDEFIWalletAdapterConfig = {}) { + super(); + this._connecting = false; + this._wallet = null; + this._publicKey = null; + + if (this._readyState !== WalletReadyState.Unsupported) { + scopePollingDetectionStrategy(() => { + if (window.xfi?.solana?.isXDEFI) { + this._readyState = WalletReadyState.Installed; + this.emit("readyStateChange", this._readyState); + return true; + } + return false; + }); + } + } + + get publicKey() { + return this._publicKey; + } + + get connecting() { + return this._connecting; + } + + get connected() { + return !!this._wallet?.isConnected; + } + + get readyState() { + return this._readyState; + } + + async connect(): Promise { + try { + if (this.connected || this.connecting) return; + if (this._readyState !== WalletReadyState.Installed) + throw new WalletNotReadyError(); + + this._connecting = true; + + // eslint-disable-next-line @typescript-eslint/no-non-null-assertion + const wallet = window.xfi!.solana!; + + if (!wallet.isConnected) { + try { + await wallet.connect(); + } catch (error: any) { + throw new WalletConnectionError(error?.message, error); + } + } + + if (!wallet.publicKey) throw new WalletAccountError(); + + let publicKey: PublicKey; + try { + publicKey = new PublicKey(wallet.publicKey.toBytes()); + } catch (error: any) { + throw new WalletPublicKeyError(error?.message, error); + } + + wallet.on("disconnect", this._disconnected); + + this._wallet = wallet; + this._publicKey = publicKey; + + this.emit("connect", publicKey); + } catch (error: any) { + this.emit("error", error); + throw error; + } finally { + this._connecting = false; + } + } + + async disconnect(): Promise { + const wallet = this._wallet; + if (wallet) { + wallet.off("disconnect", this._disconnected); + + this._wallet = null; + this._publicKey = null; + + try { + await wallet.disconnect(); + } catch (error: any) { + this.emit("error", new WalletDisconnectionError(error?.message, error)); + } + } + + this.emit("disconnect"); + } + + async signTransaction(transaction: T): Promise { + try { + const wallet = this._wallet; + if (!wallet) throw new WalletNotConnectedError(); + + try { + return ( + ((await wallet.signTransaction(transaction)) as T) || transaction + ); + } catch (error: any) { + throw new WalletSignTransactionError(error?.message, error); + } + } catch (error: any) { + this.emit("error", error); + throw error; + } + } + + async signAllTransactions( + transactions: T[], + ): Promise { + try { + const wallet = this._wallet; + if (!wallet) throw new WalletNotConnectedError(); + + try { + return ( + ((await wallet.signAllTransactions(transactions)) as T[]) || + transactions + ); + } catch (error: any) { + throw new WalletSignTransactionError(error?.message, error); + } + } catch (error: any) { + this.emit("error", error); + throw error; + } + } + + async signMessage(message: Uint8Array): Promise { + try { + const wallet = this._wallet; + if (!wallet) throw new WalletNotConnectedError(); + + try { + const { signature } = await wallet.signMessage(message); + return signature; + } catch (error: any) { + throw new WalletSignMessageError(error?.message, error); + } + } catch (error: any) { + this.emit("error", error); + throw error; + } + } + + private _disconnected = () => { + const wallet = this._wallet; + if (wallet) { + wallet.off("disconnect", this._disconnected); + + this._wallet = null; + this._publicKey = null; + + this.emit("error", new WalletDisconnectedError()); + this.emit("disconnect"); + } + }; +} +```