diff --git a/.changeset/wicked-bananas-check.md b/.changeset/wicked-bananas-check.md new file mode 100644 index 00000000..60d37d28 --- /dev/null +++ b/.changeset/wicked-bananas-check.md @@ -0,0 +1,5 @@ +--- +'@wallet-standard/react-core': major +--- + +Replaced the feature context providers with two hooks: `useConnect()` and `useDisconnect()` diff --git a/packages/react/core/README.md b/packages/react/core/README.md index b447e8a0..7a04962c 100644 --- a/packages/react/core/README.md +++ b/packages/react/core/README.md @@ -7,3 +7,45 @@ This package provides React hooks for using Wallet Standard wallets, accounts, a ### `useWallets()` Vends an array of `UiWallet` objects; one for every registered Wallet Standard `Wallet`. + +### `useConnect(uiWallet)` + +Given a `UiWallet`, this hook returns a function you can call to ask the wallet to authorize the current domain to use new accounts. Calling the returned method will return a promise for the array of account authorized by the wallet. + +```tsx +function ConnectButton({ onAccountsConnected, wallet }) { + const [isConnecting, connect] = useConnect(wallet); + return ( + + ); +} +``` + +Be aware that calls to `connect` operate on a single promise that is global to the entire app. If you call `connect` multiple times or from multiple places in your app, the exact same promise will be vended to all callers. This reflects the fact that a wallet can only field one authorization request at a time. + +### `useDisconnect(uiWallet)` + +Given a `UiWallet`, this hook returns a function you can call to ask the wallet to deauthorize accounts authorized for the current domain, or at least to remove them from the wallet's `accounts` array for the time being, depending on the wallet's implementation of `standard:disconnect`. Call the returned method will return a promise that resolves once the disconnection is complete. + +```tsx +function DisconnectButton({ onDisconnected, wallet }) { + const [isDisconnecting, disconnect] = useDisconnect(wallet); + return ( + + ); +} +``` diff --git a/packages/react/core/src/WalletStandardProvider.tsx b/packages/react/core/src/WalletStandardProvider.tsx index e0c5eaf3..22a0d5eb 100644 --- a/packages/react/core/src/WalletStandardProvider.tsx +++ b/packages/react/core/src/WalletStandardProvider.tsx @@ -1,12 +1,5 @@ import type { FC, ReactNode } from 'react'; import React from 'react'; -import { - ConnectProvider, - DisconnectProvider, - SignAndSendTransactionProvider, - SignMessageProvider, - SignTransactionProvider, -} from './features/index.js'; import { WalletAccountProvider } from './WalletAccountProvider.js'; import { WalletProvider } from './WalletProvider.js'; @@ -17,20 +10,10 @@ export interface WalletStandardProviderProps { } /** TODO: docs */ -export const WalletStandardProvider: FC = ({ children, onError }) => { +export const WalletStandardProvider: FC = ({ children }) => { return ( - - - - - - {children} - - - - - + {children} ); }; diff --git a/packages/react/core/src/__tests__/useConnect-test.ts b/packages/react/core/src/__tests__/useConnect-test.ts new file mode 100644 index 00000000..0b279e7e --- /dev/null +++ b/packages/react/core/src/__tests__/useConnect-test.ts @@ -0,0 +1,130 @@ +import type { Wallet, WalletVersion } from '@wallet-standard/base'; +import { StandardConnect } from '@wallet-standard/features'; +import type { UiWallet } from '@wallet-standard/ui'; +import { getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED } from '@wallet-standard/ui-registry'; + +import { useConnect } from '../features/useConnect.js'; +import { renderHook } from '../test-renderer.js'; + +jest.mock('@wallet-standard/ui-registry'); + +describe('useConnect', () => { + let mockConnect: jest.Mock; + let mockUiWallet: UiWallet; + beforeEach(() => { + mockConnect = jest.fn().mockResolvedValue({ accounts: [] }); + const mockWallet: Wallet = { + accounts: [], + chains: [], + features: { + [StandardConnect]: { + connect: mockConnect, + }, + }, + icon: '', + name: 'Mock Wallet', + version: '1.0.0' as WalletVersion, + }; + jest.mocked(getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED).mockReturnValue(mockWallet); + // Suppresses console output when an `ErrorBoundary` is hit. + // See https://stackoverflow.com/a/72632884/802047 + jest.spyOn(console, 'error').mockImplementation(); + jest.spyOn(console, 'warn').mockImplementation(); + }); + describe('the `isConnecting` property', () => { + it('is `false` before calling `connect()`', () => { + const { result } = renderHook(() => useConnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } + const [isConnecting, _] = result.current; + expect(isConnecting).toBe(false); + }); + it('is `false` after the connection resolves', async () => { + expect.assertions(1); + const { result } = renderHook(() => useConnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } + const [_A, connect] = result.current; + await connect(); + const [isConnecting, _B] = result.current; + expect(isConnecting).toBe(false); + }); + it('is `true` after calling `connect()`', () => { + const { result } = renderHook(() => useConnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } + const [_, connect] = result.current; + connect(); + const [isConnecting] = result.current; + expect(isConnecting).toBe(true); + }); + it('is `true` on hooks that did not trigger the connect', () => { + const { result: resultA } = renderHook(() => useConnect(mockUiWallet)); + const { result: resultB } = renderHook(() => useConnect(mockUiWallet)); + if (resultA.__type === 'error' || !resultA.current) { + throw resultA.current; + } + if (resultB.__type === 'error' || !resultB.current) { + throw resultB.current; + } + const [_, connectA] = resultA.current; + connectA(); + const [isConnectingB] = resultB.current; + expect(isConnectingB).toBe(true); + }); + }); + describe('the `connect` property', () => { + it("calls the wallet's connect implementation when called ", () => { + const { result } = renderHook(() => useConnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } else { + const [_, connect] = result.current; + connect(); + expect(mockConnect).toHaveBeenCalled(); + } + }); + it("calls the wallet's connect implementation once despite multiple calls", () => { + const { result } = renderHook(() => useConnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } else { + const [_, connect] = result.current; + connect(); + connect(); + expect(mockConnect).toHaveBeenCalledTimes(1); + } + }); + it("calls the wallet's connect implementation once despite calls from different hooks", () => { + const { result: resultA } = renderHook(() => useConnect(mockUiWallet)); + const { result: resultB } = renderHook(() => useConnect(mockUiWallet)); + if (resultA.__type === 'error' || !resultA.current) { + throw resultA.current; + } else if (resultB.__type === 'error' || !resultB.current) { + throw resultB.current; + } else { + const [_A, connectA] = resultA.current; + const [_B, connectB] = resultB.current; + connectA(); + connectB(); + expect(mockConnect).toHaveBeenCalledTimes(1); + } + }); + it("calls the wallet's connect implementation anew after the first connection resolves", async () => { + expect.assertions(1); + const { result } = renderHook(() => useConnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } else { + const [_, connect] = result.current; + connect(); + await jest.runAllTimersAsync(); + connect(); + expect(mockConnect).toHaveBeenCalledTimes(2); + } + }); + }); +}); diff --git a/packages/react/core/src/__tests__/useDisconnect-test.ts b/packages/react/core/src/__tests__/useDisconnect-test.ts new file mode 100644 index 00000000..fddb072d --- /dev/null +++ b/packages/react/core/src/__tests__/useDisconnect-test.ts @@ -0,0 +1,130 @@ +import type { Wallet, WalletVersion } from '@wallet-standard/base'; +import { StandardDisconnect } from '@wallet-standard/features'; +import type { UiWallet } from '@wallet-standard/ui'; +import { getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED } from '@wallet-standard/ui-registry'; + +import { useDisconnect } from '../features/useDisconnect.js'; +import { renderHook } from '../test-renderer.js'; + +jest.mock('@wallet-standard/ui-registry'); + +describe('useDisconnect', () => { + let mockDisconnect: jest.Mock; + let mockUiWallet: UiWallet; + beforeEach(() => { + mockDisconnect = jest.fn().mockResolvedValue({ accounts: [] }); + const mockWallet: Wallet = { + accounts: [], + chains: [], + features: { + [StandardDisconnect]: { + disconnect: mockDisconnect, + }, + }, + icon: '', + name: 'Mock Wallet', + version: '1.0.0' as WalletVersion, + }; + jest.mocked(getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED).mockReturnValue(mockWallet); + // Suppresses console output when an `ErrorBoundary` is hit. + // See https://stackoverflow.com/a/72632884/802047 + jest.spyOn(console, 'error').mockImplementation(); + jest.spyOn(console, 'warn').mockImplementation(); + }); + describe('the `isDisconnecting` property', () => { + it('is `false` before calling `disconnect()`', () => { + const { result } = renderHook(() => useDisconnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } + const [isDisconnecting, _] = result.current; + expect(isDisconnecting).toBe(false); + }); + it('is `false` after the disconnection resolves', async () => { + expect.assertions(1); + const { result } = renderHook(() => useDisconnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } + const [_A, disconnectA] = result.current; + await disconnectA(); + const [isDisconnecting, _B] = result.current; + expect(isDisconnecting).toBe(false); + }); + it('is `true` after calling `disconnect()`', () => { + const { result } = renderHook(() => useDisconnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } + const [_, disconnect] = result.current; + disconnect(); + const [isDisconnecting] = result.current; + expect(isDisconnecting).toBe(true); + }); + it('is `true` on hooks that did not trigger the disconnect', () => { + const { result: resultA } = renderHook(() => useDisconnect(mockUiWallet)); + const { result: resultB } = renderHook(() => useDisconnect(mockUiWallet)); + if (resultA.__type === 'error' || !resultA.current) { + throw resultA.current; + } + if (resultB.__type === 'error' || !resultB.current) { + throw resultB.current; + } + const [_, disconnectA] = resultA.current; + disconnectA(); + const [isDisconnectingB] = resultB.current; + expect(isDisconnectingB).toBe(true); + }); + }); + describe('the `disconnect` property', () => { + it("calls the wallet's disconnect implementation when called ", () => { + const { result } = renderHook(() => useDisconnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } else { + const [_, disconnect] = result.current; + disconnect(); + expect(mockDisconnect).toHaveBeenCalled(); + } + }); + it("calls the wallet's disconnect implementation once despite multiple calls", () => { + const { result } = renderHook(() => useDisconnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } else { + const [_, disconnect] = result.current; + disconnect(); + disconnect(); + expect(mockDisconnect).toHaveBeenCalledTimes(1); + } + }); + it("calls the wallet's disconnect implementation once despite calls from different hooks", () => { + const { result: resultA } = renderHook(() => useDisconnect(mockUiWallet)); + const { result: resultB } = renderHook(() => useDisconnect(mockUiWallet)); + if (resultA.__type === 'error' || !resultA.current) { + throw resultA.current; + } else if (resultB.__type === 'error' || !resultB.current) { + throw resultB.current; + } else { + const [_A, disconnectA] = resultA.current; + const [_B, disconnectB] = resultB.current; + disconnectA(); + disconnectB(); + expect(mockDisconnect).toHaveBeenCalledTimes(1); + } + }); + it("calls the wallet's disconnect implementation anew after the first disconnection resolves", async () => { + expect.assertions(1); + const { result } = renderHook(() => useDisconnect(mockUiWallet)); + if (result.__type === 'error' || !result.current) { + throw result.current; + } else { + const [_, disconnect] = result.current; + disconnect(); + await jest.runAllTimersAsync(); + disconnect(); + expect(mockDisconnect).toHaveBeenCalledTimes(2); + } + }); + }); +}); diff --git a/packages/react/core/src/__tests__/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT-test.ts b/packages/react/core/src/__tests__/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT-test.ts index db27c4fb..84e8732e 100644 --- a/packages/react/core/src/__tests__/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT-test.ts +++ b/packages/react/core/src/__tests__/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT-test.ts @@ -21,8 +21,6 @@ describe('useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT', () => { on: mockOn, register: mockRegister, }); - // Suppresses console output when an `ErrorBoundary` is hit. - // See https://stackoverflow.com/a/72632884/802047 }); it('returns a list of registered wallets', () => { const expectedWallets = [] as readonly Wallet[]; diff --git a/packages/react/core/src/features/connect/ConnectProvider.tsx b/packages/react/core/src/features/connect/ConnectProvider.tsx deleted file mode 100644 index a11d9da3..00000000 --- a/packages/react/core/src/features/connect/ConnectProvider.tsx +++ /dev/null @@ -1,80 +0,0 @@ -import type { Wallet } from '@wallet-standard/base'; -import type { ConnectFeature, ConnectMethod } from '@wallet-standard/features'; -import type { FC, ReactNode } from 'react'; -import React, { useCallback, useMemo, useRef, useState } from 'react'; -import { useWallet } from '../../useWallet.js'; -import { ConnectContext } from './useConnect.js'; - -/** TODO: docs */ -export interface ConnectProviderProps { - children: NonNullable; - onError?: (error: Error) => void; -} - -/** TODO: docs */ -export function hasConnectFeature(features: Wallet['features']): features is ConnectFeature { - return 'standard:connect' in features; -} - -/** TODO: docs */ -export const ConnectProvider: FC = ({ children, onError }) => { - const { features } = useWallet(); - - // Handle errors, logging them by default. - const handleError = useCallback( - (error: Error) => { - (onError || console.error)(error); - return error; - }, - [onError] - ); - - // Connect to the wallet. - const [waiting, setWaiting] = useState(false); - const promise = useRef>(); - const connect = useMemo( - () => - hasConnectFeature(features) - ? async (input) => { - // FIXME: if called first with silent=true, promise.current will be set but waiting will be false - - // If already waiting, wait for that promise to resolve. - if (promise.current) { - try { - await promise.current; - } catch (error: any) { - // Error will already have been handled below. - } - } - - const { silent } = input || {}; - if (!silent) { - setWaiting(true); - } - try { - promise.current = features['standard:connect'].connect(input); - return await promise.current; - } catch (error: any) { - throw handleError(error); - } finally { - if (!silent) { - setWaiting(false); - } - promise.current = undefined; - } - } - : undefined, - [features, promise, handleError] - ); - - return ( - - {children} - - ); -}; diff --git a/packages/react/core/src/features/connect/index.ts b/packages/react/core/src/features/connect/index.ts deleted file mode 100644 index 8148b972..00000000 --- a/packages/react/core/src/features/connect/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export * from './ConnectProvider.js'; -export * from './useConnect.js'; diff --git a/packages/react/core/src/features/connect/useConnect.ts b/packages/react/core/src/features/connect/useConnect.ts deleted file mode 100644 index 0809ceb5..00000000 --- a/packages/react/core/src/features/connect/useConnect.ts +++ /dev/null @@ -1,24 +0,0 @@ -import type { ConnectMethod } from '@wallet-standard/features'; -import { createContext, useContext } from 'react'; -import { createDefaultContext } from '../../context.js'; - -/** TODO: docs */ -export interface ConnectContextState { - waiting: boolean; - connect: ConnectMethod | undefined; -} - -const DEFAULT_CONNECT_STATE: Readonly = { - waiting: false, - connect: undefined, -} as const; - -const DEFAULT_CONNECT_CONTEXT = createDefaultContext('Connect', DEFAULT_CONNECT_STATE); - -/** TODO: docs */ -export const ConnectContext = createContext(DEFAULT_CONNECT_CONTEXT); - -/** TODO: docs */ -export function useConnect(): ConnectContextState { - return useContext(ConnectContext); -} diff --git a/packages/react/core/src/features/disconnect/DisconnectProvider.tsx b/packages/react/core/src/features/disconnect/DisconnectProvider.tsx deleted file mode 100644 index d5ea9c0c..00000000 --- a/packages/react/core/src/features/disconnect/DisconnectProvider.tsx +++ /dev/null @@ -1,73 +0,0 @@ -import type { Wallet } from '@wallet-standard/base'; -import type { DisconnectFeature, DisconnectMethod } from '@wallet-standard/features'; -import type { FC, ReactNode } from 'react'; -import React, { useCallback, useMemo, useRef, useState } from 'react'; -import { useWallet } from '../../useWallet.js'; -import { DisconnectContext } from './useDisconnect.js'; - -/** TODO: docs */ -export interface DisconnectProviderProps { - children: NonNullable; - onError?: (error: Error) => void; -} - -/** TODO: docs */ -export function hasDisconnectFeature(features: Wallet['features']): features is DisconnectFeature { - return 'standard:disconnect' in features; -} - -/** TODO: docs */ -export const DisconnectProvider: FC = ({ children, onError }) => { - const { features } = useWallet(); - - // Handle errors, logging them by default. - const handleError = useCallback( - (error: Error) => { - (onError || console.error)(error); - return error; - }, - [onError] - ); - - // Disconnect from the wallet. - const [waiting, setWaiting] = useState(false); - const promise = useRef>(); - const disconnect = useMemo( - () => - hasDisconnectFeature(features) - ? async () => { - // If already waiting, wait for that promise to resolve. - if (promise.current) { - try { - await promise.current; - } catch (error: any) { - // Error will already have been handled below. - } - } - - setWaiting(true); - try { - promise.current = features['standard:disconnect'].disconnect(); - return await promise.current; - } catch (error: any) { - throw handleError(error); - } finally { - setWaiting(false); - promise.current = undefined; - } - } - : undefined, - [features, promise, handleError] - ); - - return ( - - {children} - - ); -}; diff --git a/packages/react/core/src/features/disconnect/index.ts b/packages/react/core/src/features/disconnect/index.ts deleted file mode 100644 index 3b52d541..00000000 --- a/packages/react/core/src/features/disconnect/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export * from './DisconnectProvider.js'; -export * from './useDisconnect.js'; diff --git a/packages/react/core/src/features/disconnect/useDisconnect.ts b/packages/react/core/src/features/disconnect/useDisconnect.ts deleted file mode 100644 index e5f9bec5..00000000 --- a/packages/react/core/src/features/disconnect/useDisconnect.ts +++ /dev/null @@ -1,24 +0,0 @@ -import type { DisconnectMethod } from '@wallet-standard/features'; -import { createContext, useContext } from 'react'; -import { createDefaultContext } from '../../context.js'; - -/** TODO: docs */ -export interface DisconnectContextState { - waiting: boolean; - disconnect: DisconnectMethod | undefined; -} - -const DEFAULT_DISCONNECT_STATE: Readonly = { - waiting: false, - disconnect: undefined, -} as const; - -const DEFAULT_DISCONNECT_CONTEXT = createDefaultContext('Disconnect', DEFAULT_DISCONNECT_STATE); - -/** TODO: docs */ -export const DisconnectContext = createContext(DEFAULT_DISCONNECT_CONTEXT); - -/** TODO: docs */ -export function useDisconnect(): DisconnectContextState { - return useContext(DisconnectContext); -} diff --git a/packages/react/core/src/features/events.ts b/packages/react/core/src/features/events.ts new file mode 100644 index 00000000..4b33dbec --- /dev/null +++ b/packages/react/core/src/features/events.ts @@ -0,0 +1,7 @@ +import type { Wallet, WalletWithFeatures } from '@wallet-standard/base'; +import type { StandardEventsFeature } from '@wallet-standard/features'; +import { StandardEvents } from '@wallet-standard/features'; + +export function walletHasStandardEventsFeature(wallet: Wallet): wallet is WalletWithFeatures { + return StandardEvents in wallet.features; +} diff --git a/packages/react/core/src/features/index.ts b/packages/react/core/src/features/index.ts index 42c05024..69b7b52b 100644 --- a/packages/react/core/src/features/index.ts +++ b/packages/react/core/src/features/index.ts @@ -1,5 +1,2 @@ -export * from './connect/index.js'; -export * from './disconnect/index.js'; -export * from './signAndSendTransaction/index.js'; -export * from './signMessage/index.js'; -export * from './signTransaction/index.js'; +export * from './useConnect.js'; +export * from './useDisconnect.js'; diff --git a/packages/react/core/src/features/signAndSendTransaction/SignAndSendTransactionProvider.tsx b/packages/react/core/src/features/signAndSendTransaction/SignAndSendTransactionProvider.tsx deleted file mode 100644 index 5db0fc47..00000000 --- a/packages/react/core/src/features/signAndSendTransaction/SignAndSendTransactionProvider.tsx +++ /dev/null @@ -1,80 +0,0 @@ -import type { Wallet } from '@wallet-standard/base'; -import type { - SignAndSendTransactionFeature, - SignAndSendTransactionMethod, -} from '@wallet-standard/experimental-features'; -import type { FC, ReactNode } from 'react'; -import React, { useCallback, useMemo, useRef, useState } from 'react'; -import { useWallet } from '../../useWallet.js'; -import { SignAndSendTransactionContext } from './useSignAndSendTransaction.js'; - -/** TODO: docs */ -export interface SignAndSendTransactionProviderProps { - children: NonNullable; - onError?: (error: Error) => void; -} - -/** TODO: docs */ -export function hasSignAndSendTransactionFeature( - features: Wallet['features'] -): features is SignAndSendTransactionFeature { - return 'experimental:signAndSendTransaction' in features; -} - -/** TODO: docs */ -export const SignAndSendTransactionProvider: FC = ({ children, onError }) => { - const { features } = useWallet(); - - // Handle errors, logging them by default. - const handleError = useCallback( - (error: Error) => { - (onError || console.error)(error); - return error; - }, - [onError] - ); - - // Sign transactions with the wallet. - const [waiting, setWaiting] = useState(false); - const promise = useRef>(); - const signAndSendTransaction = useMemo( - () => - hasSignAndSendTransactionFeature(features) - ? async (...inputs) => { - // If already waiting, wait for that promise to resolve. - if (promise.current) { - try { - await promise.current; - } catch (error: any) { - // Error will already have been handled below. - } - } - - setWaiting(true); - try { - promise.current = features['experimental:signAndSendTransaction'].signAndSendTransaction( - ...inputs - ); - return await promise.current; - } catch (error: any) { - throw handleError(error); - } finally { - setWaiting(false); - promise.current = undefined; - } - } - : undefined, - [features, promise, handleError] - ); - - return ( - - {children} - - ); -}; diff --git a/packages/react/core/src/features/signAndSendTransaction/index.ts b/packages/react/core/src/features/signAndSendTransaction/index.ts deleted file mode 100644 index f1ad145a..00000000 --- a/packages/react/core/src/features/signAndSendTransaction/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export * from './SignAndSendTransactionProvider.js'; -export * from './useSignAndSendTransaction.js'; diff --git a/packages/react/core/src/features/signAndSendTransaction/useSignAndSendTransaction.ts b/packages/react/core/src/features/signAndSendTransaction/useSignAndSendTransaction.ts deleted file mode 100644 index 26510940..00000000 --- a/packages/react/core/src/features/signAndSendTransaction/useSignAndSendTransaction.ts +++ /dev/null @@ -1,27 +0,0 @@ -import type { SignAndSendTransactionMethod } from '@wallet-standard/experimental-features'; -import { createContext, useContext } from 'react'; -import { createDefaultContext } from '../../context.js'; - -/** TODO: docs */ -export interface SignAndSendTransactionContextState { - waiting: boolean; - signAndSendTransaction: SignAndSendTransactionMethod | undefined; -} - -const DEFAULT_SIGN_AND_SEND_TRANSACTION_STATE: Readonly = { - waiting: false, - signAndSendTransaction: undefined, -} as const; - -const DEFAULT_SIGN_AND_SEND_TRANSACTION_CONTEXT = createDefaultContext( - 'SignAndSendTransaction', - DEFAULT_SIGN_AND_SEND_TRANSACTION_STATE -); - -/** TODO: docs */ -export const SignAndSendTransactionContext = createContext(DEFAULT_SIGN_AND_SEND_TRANSACTION_CONTEXT); - -/** TODO: docs */ -export function useSignAndSendTransaction(): SignAndSendTransactionContextState { - return useContext(SignAndSendTransactionContext); -} diff --git a/packages/react/core/src/features/signMessage/SignMessageProvider.tsx b/packages/react/core/src/features/signMessage/SignMessageProvider.tsx deleted file mode 100644 index acae7b53..00000000 --- a/packages/react/core/src/features/signMessage/SignMessageProvider.tsx +++ /dev/null @@ -1,73 +0,0 @@ -import type { Wallet } from '@wallet-standard/base'; -import type { SignMessageFeature, SignMessageMethod } from '@wallet-standard/experimental-features'; -import type { FC, ReactNode } from 'react'; -import React, { useCallback, useMemo, useRef, useState } from 'react'; -import { useWallet } from '../../useWallet.js'; -import { SignMessageContext } from './useSignMessage.js'; - -/** TODO: docs */ -export interface SignMessageProviderProps { - children: NonNullable; - onError?: (error: Error) => void; -} - -/** TODO: docs */ -export function hasSignMessageFeature(features: Wallet['features']): features is SignMessageFeature { - return 'experimental:signMessage' in features; -} - -/** TODO: docs */ -export const SignMessageProvider: FC = ({ children, onError }) => { - const { features } = useWallet(); - - // Handle errors, logging them by default. - const handleError = useCallback( - (error: Error) => { - (onError || console.error)(error); - return error; - }, - [onError] - ); - - // Sign messages with the wallet. - const [waiting, setWaiting] = useState(false); - const promise = useRef>(); - const signMessage = useMemo( - () => - hasSignMessageFeature(features) - ? async (...inputs) => { - // If already waiting, wait for that promise to resolve. - if (promise.current) { - try { - await promise.current; - } catch (error: any) { - // Error will already have been handled below. - } - } - - setWaiting(true); - try { - promise.current = features['experimental:signMessage'].signMessage(...inputs); - return await promise.current; - } catch (error: any) { - throw handleError(error); - } finally { - setWaiting(false); - promise.current = undefined; - } - } - : undefined, - [features, promise, handleError] - ); - - return ( - - {children} - - ); -}; diff --git a/packages/react/core/src/features/signMessage/index.ts b/packages/react/core/src/features/signMessage/index.ts deleted file mode 100644 index 7996c450..00000000 --- a/packages/react/core/src/features/signMessage/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export * from './SignMessageProvider.js'; -export * from './useSignMessage.js'; diff --git a/packages/react/core/src/features/signMessage/useSignMessage.ts b/packages/react/core/src/features/signMessage/useSignMessage.ts deleted file mode 100644 index 9a900781..00000000 --- a/packages/react/core/src/features/signMessage/useSignMessage.ts +++ /dev/null @@ -1,24 +0,0 @@ -import type { SignMessageMethod } from '@wallet-standard/experimental-features'; -import { createContext, useContext } from 'react'; -import { createDefaultContext } from '../../context.js'; - -/** TODO: docs */ -export interface SignMessageContextState { - waiting: boolean; - signMessage: SignMessageMethod | undefined; -} - -const DEFAULT_SIGN_MESSAGE_STATE: Readonly = { - waiting: false, - signMessage: undefined, -} as const; - -const DEFAULT_SIGN_MESSAGE_CONTEXT = createDefaultContext('SignMessage', DEFAULT_SIGN_MESSAGE_STATE); - -/** TODO: docs */ -export const SignMessageContext = createContext(DEFAULT_SIGN_MESSAGE_CONTEXT); - -/** TODO: docs */ -export function useSignMessage(): SignMessageContextState { - return useContext(SignMessageContext); -} diff --git a/packages/react/core/src/features/signTransaction/SignTransactionProvider.tsx b/packages/react/core/src/features/signTransaction/SignTransactionProvider.tsx deleted file mode 100644 index 1d115119..00000000 --- a/packages/react/core/src/features/signTransaction/SignTransactionProvider.tsx +++ /dev/null @@ -1,73 +0,0 @@ -import type { Wallet } from '@wallet-standard/base'; -import type { SignTransactionFeature, SignTransactionMethod } from '@wallet-standard/experimental-features'; -import type { FC, ReactNode } from 'react'; -import React, { useCallback, useMemo, useRef, useState } from 'react'; -import { useWallet } from '../../useWallet.js'; -import { SignTransactionContext } from './useSignTransaction.js'; - -/** TODO: docs */ -export interface SignTransactionProviderProps { - children: NonNullable; - onError?: (error: Error) => void; -} - -/** TODO: docs */ -export function hasSignTransactionFeature(features: Wallet['features']): features is SignTransactionFeature { - return 'experimental:signTransaction' in features; -} - -/** TODO: docs */ -export const SignTransactionProvider: FC = ({ children, onError }) => { - const { features } = useWallet(); - - // Handle errors, logging them by default. - const handleError = useCallback( - (error: Error) => { - (onError || console.error)(error); - return error; - }, - [onError] - ); - - // Sign transactions with the wallet. - const [waiting, setWaiting] = useState(false); - const promise = useRef>(); - const signTransaction = useMemo( - () => - hasSignTransactionFeature(features) - ? async (...inputs) => { - // If already waiting, wait for that promise to resolve. - if (promise.current) { - try { - await promise.current; - } catch (error: any) { - // Error will already have been handled below. - } - } - - setWaiting(true); - try { - promise.current = features['experimental:signTransaction'].signTransaction(...inputs); - return await promise.current; - } catch (error: any) { - throw handleError(error); - } finally { - setWaiting(false); - promise.current = undefined; - } - } - : undefined, - [features, promise, handleError] - ); - - return ( - - {children} - - ); -}; diff --git a/packages/react/core/src/features/signTransaction/index.ts b/packages/react/core/src/features/signTransaction/index.ts deleted file mode 100644 index afab8faa..00000000 --- a/packages/react/core/src/features/signTransaction/index.ts +++ /dev/null @@ -1,2 +0,0 @@ -export * from './SignTransactionProvider.js'; -export * from './useSignTransaction.js'; diff --git a/packages/react/core/src/features/signTransaction/useSignTransaction.ts b/packages/react/core/src/features/signTransaction/useSignTransaction.ts deleted file mode 100644 index cb5d49c9..00000000 --- a/packages/react/core/src/features/signTransaction/useSignTransaction.ts +++ /dev/null @@ -1,24 +0,0 @@ -import type { SignTransactionMethod } from '@wallet-standard/experimental-features'; -import { createContext, useContext } from 'react'; -import { createDefaultContext } from '../../context.js'; - -/** TODO: docs */ -export interface SignTransactionContextState { - waiting: boolean; - signTransaction: SignTransactionMethod | undefined; -} - -const DEFAULT_SIGN_TRANSACTION_STATE: Readonly = { - waiting: false, - signTransaction: undefined, -} as const; - -const DEFAULT_SIGN_TRANSACTION_CONTEXT = createDefaultContext('SignTransaction', DEFAULT_SIGN_TRANSACTION_STATE); - -/** TODO: docs */ -export const SignTransactionContext = createContext(DEFAULT_SIGN_TRANSACTION_CONTEXT); - -/** TODO: docs */ -export function useSignTransaction(): SignTransactionContextState { - return useContext(SignTransactionContext); -} diff --git a/packages/react/core/src/features/useConnect.ts b/packages/react/core/src/features/useConnect.ts new file mode 100644 index 00000000..19add354 --- /dev/null +++ b/packages/react/core/src/features/useConnect.ts @@ -0,0 +1,66 @@ +import type { StandardConnectFeature, StandardConnectInput, StandardConnectMethod } from '@wallet-standard/features'; +import { StandardConnect } from '@wallet-standard/features'; +import type { UiWallet, UiWalletAccount } from '@wallet-standard/ui'; +import { getWalletFeature } from '@wallet-standard/ui'; +import { + getOrCreateUiWalletAccountForStandardWalletAccount_DO_NOT_USE_OR_YOU_WILL_BE_FIRED, + getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED, +} from '@wallet-standard/ui-registry'; +import { useCallback } from 'react'; + +import { useWeakRef } from '../useWeakRef.js'; + +type TupleSplit = N extends T['length'] + ? [T, []] + : [T extends [...infer U, ...infer R] ? [...U, ...TupleSplit[0]] : never, TupleSplit[1]]; + +type SkipFirst = T extends [] ? [] : TupleSplit[1]; + +/** + * Returns a function you can call to ask the wallet to authorize the current domain to use new + * accounts. + */ +export function useConnect( + uiWallet: TWallet +): [ + isConnecting: boolean, + connect: ( + input?: Omit[0]>, 'silent'>, + ...rest: SkipFirst, 1> + ) => Promise, +] { + const connectFeature = getWalletFeature( + uiWallet, + StandardConnect + ) as StandardConnectFeature[typeof StandardConnect]; + const wallet = getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED(uiWallet); + const connectionPromise = useWeakRef>(wallet); + const connect = useCallback( + async function connect( + input?: Omit[0]>, 'silent'>, + ...rest: SkipFirst, 1> + ) { + if (connectionPromise.current) { + return connectionPromise.current; + } + const newConnectionPromise = connectFeature + .connect(input, ...rest) + .then(({ accounts }) => { + return accounts.map( + getOrCreateUiWalletAccountForStandardWalletAccount_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.bind( + null, + wallet + ) + ) as readonly UiWalletAccount[]; + }) + .finally(() => { + connectionPromise.current = undefined; + }); + connectionPromise.current = newConnectionPromise; + return await newConnectionPromise; + }, + [connectFeature, connectionPromise, wallet] + ); + const isConnecting = !!connectionPromise.current; + return [isConnecting, connect]; +} diff --git a/packages/react/core/src/features/useDisconnect.ts b/packages/react/core/src/features/useDisconnect.ts new file mode 100644 index 00000000..7e1e736f --- /dev/null +++ b/packages/react/core/src/features/useDisconnect.ts @@ -0,0 +1,39 @@ +import type { StandardDisconnectFeature, StandardDisconnectMethod } from '@wallet-standard/features'; +import { StandardDisconnect } from '@wallet-standard/features'; +import type { UiWallet } from '@wallet-standard/ui'; +import { getWalletFeature } from '@wallet-standard/ui'; +import { getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED } from '@wallet-standard/ui-registry'; +import { useCallback } from 'react'; + +import { useWeakRef } from '../useWeakRef.js'; + +/** + * Returns a function you can call to ask the wallet to deauthorize accounts authorized for the + * current domain, or at least to remove them from the wallet's `accounts` array for the time being, + * depending on the wallet's implementation of `standard:disconnect`. + */ +export function useDisconnect( + uiWallet: TWallet +): [isDisconnecting: boolean, disconnect: (...inputs: Parameters) => Promise] { + const disconnectFeature = getWalletFeature( + uiWallet, + StandardDisconnect + ) as StandardDisconnectFeature[typeof StandardDisconnect]; + const wallet = getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED(uiWallet); + const disconnectionPromise = useWeakRef | undefined>(wallet); + const disconnect = useCallback( + async function disconnect(...inputs: Parameters) { + if (disconnectionPromise.current) { + return disconnectionPromise.current; + } + const newDisconnectionPromise = disconnectFeature.disconnect(...inputs).finally(() => { + disconnectionPromise.current = undefined; + }); + disconnectionPromise.current = newDisconnectionPromise; + return await newDisconnectionPromise; + }, + [disconnectFeature, disconnectionPromise] + ); + const isDisconnecting = !!disconnectionPromise.current; + return [isDisconnecting, disconnect]; +} diff --git a/packages/react/core/src/features/version.ts b/packages/react/core/src/features/version.ts new file mode 100644 index 00000000..42aaf466 --- /dev/null +++ b/packages/react/core/src/features/version.ts @@ -0,0 +1,3 @@ +import type { WalletVersion } from '@wallet-standard/base'; + +export const FEATURE_HOOKS_SUPPORTED_WALLET_VERSION = '1.0.0' as WalletVersion; diff --git a/packages/react/core/src/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT.ts b/packages/react/core/src/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT.ts index 3b7cfcd6..4b265cae 100644 --- a/packages/react/core/src/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT.ts +++ b/packages/react/core/src/useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT.ts @@ -1,9 +1,9 @@ import { getWallets } from '@wallet-standard/app'; -import type { Wallet, WalletWithFeatures } from '@wallet-standard/base'; -import { StandardEvents, type StandardEventsFeature } from '@wallet-standard/features'; +import type { Wallet } from '@wallet-standard/base'; +import { StandardEvents } from '@wallet-standard/features'; import { useCallback, useRef, useSyncExternalStore } from 'react'; -import { hasEventsFeature } from './WalletProvider.js'; +import { walletHasStandardEventsFeature } from './features/events.js'; import { useStable } from './useStable.js'; const NO_WALLETS: readonly Wallet[] = []; @@ -12,10 +12,6 @@ function getServerSnapshot(): readonly Wallet[] { return NO_WALLETS; } -function walletHasStandardEventsFeature(wallet: Wallet): wallet is WalletWithFeatures { - return hasEventsFeature(wallet.features); -} - /** TODO: docs */ export function useWallets_INTERNAL_ONLY_NOT_FOR_EXPORT(): readonly Wallet[] { const { get, on } = useStable(getWallets); diff --git a/packages/ui/features/src/__tests__/getWalletFeature-test.ts b/packages/ui/features/src/__tests__/getWalletFeature-test.ts index 1393e1d3..21e6555b 100644 --- a/packages/ui/features/src/__tests__/getWalletFeature-test.ts +++ b/packages/ui/features/src/__tests__/getWalletFeature-test.ts @@ -1,5 +1,8 @@ import type { Wallet, WalletVersion } from '@wallet-standard/base'; -import { WalletStandardError } from '@wallet-standard/errors'; +import { + WALLET_STANDARD_ERROR__FEATURES__WALLET_FEATURE_UNIMPLEMENTED, + WalletStandardError, +} from '@wallet-standard/errors'; import type { UiWalletHandle } from '@wallet-standard/ui-core'; import { getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED } from '@wallet-standard/ui-registry'; @@ -36,7 +39,14 @@ describe('getWalletFeature', () => { it('throws if the handle provided does not support the feature requested', () => { expect(() => { getWalletFeature(mockWalletHandle, 'feature:b'); - }).toThrow(WalletStandardError); + }).toThrow( + new WalletStandardError(WALLET_STANDARD_ERROR__FEATURES__WALLET_FEATURE_UNIMPLEMENTED, { + featureName: 'feature:b', + supportedChains: ['solana:mainnet'], + supportedFeatures: ['feature:a'], + walletName: 'Mock Wallet', + }) + ); }); it('returns the feature of the underlying wallet', () => { jest.mocked(getWalletForHandle_DO_NOT_USE_OR_YOU_WILL_BE_FIRED).mockReturnValue(mockWallet);