UNPKG

@mysten/dapp-kit

Version:

A collection of React hooks and components for interacting with the Sui blockchain and wallets.

835 lines (805 loc) 29.2 kB
var __typeError = (msg) => { throw TypeError(msg); }; var __accessCheck = (obj, member, msg) => member.has(obj) || __typeError("Cannot " + msg); var __privateGet = (obj, member, getter) => (__accessCheck(obj, member, "read from private field"), getter ? getter.call(obj) : member.get(obj)); var __privateAdd = (obj, member, value) => member.has(obj) ? __typeError("Cannot add the same private member more than once") : member instanceof WeakSet ? member.add(obj) : member.set(obj, value); // src/components/WalletProvider.tsx import { useRef } from "react"; // src/constants/walletDefaults.ts import { SLUSH_WALLET_NAME } from "@mysten/slush-wallet"; // src/utils/stateStorage.ts function createInMemoryStore() { const store = /* @__PURE__ */ new Map(); return { getItem(key) { return store.get(key); }, setItem(key, value) { store.set(key, value); }, removeItem(key) { store.delete(key); } }; } // src/constants/walletDefaults.ts var SUI_WALLET_NAME = "Sui Wallet"; var DEFAULT_STORAGE = typeof window !== "undefined" && window.localStorage ? localStorage : createInMemoryStore(); var DEFAULT_STORAGE_KEY = "sui-dapp-kit:wallet-connection-info"; var SIGN_FEATURES = [ "sui:signTransaction", "sui:signTransactionBlock" ]; var DEFAULT_WALLET_FILTER = (wallet) => SIGN_FEATURES.some((feature) => wallet.features[feature]); var DEFAULT_PREFERRED_WALLETS = [SUI_WALLET_NAME, SLUSH_WALLET_NAME]; // src/contexts/walletContext.ts import { createContext } from "react"; var WalletContext = createContext(null); // src/hooks/wallet/useAutoConnectWallet.ts import { useQuery } from "@tanstack/react-query"; import { useLayoutEffect, useState } from "react"; // src/utils/walletUtils.ts import { getWallets, isWalletWithRequiredFeatureSet } from "@mysten/wallet-standard"; function getRegisteredWallets(preferredWallets, walletFilter) { const walletsApi = getWallets(); const wallets = walletsApi.get(); const suiWallets = wallets.filter( (wallet) => isWalletWithRequiredFeatureSet(wallet) && (!walletFilter || walletFilter(wallet)) ); return [ // Preferred wallets, in order: ...preferredWallets.map((name) => suiWallets.find((wallet) => wallet.name === name)).filter(Boolean), // Wallets in default order: ...suiWallets.filter((wallet) => !preferredWallets.includes(wallet.name)) ]; } function getWalletUniqueIdentifier(wallet) { return wallet?.id ?? wallet?.name; } // src/hooks/wallet/useConnectWallet.ts import { useMutation } from "@tanstack/react-query"; // src/constants/walletMutationKeys.ts var walletMutationKeys = { all: { baseScope: "wallet" }, connectWallet: formMutationKeyFn("connect-wallet"), autoconnectWallet: formMutationKeyFn("autoconnect-wallet"), disconnectWallet: formMutationKeyFn("disconnect-wallet"), signPersonalMessage: formMutationKeyFn("sign-personal-message"), signTransaction: formMutationKeyFn("sign-transaction"), signAndExecuteTransaction: formMutationKeyFn("sign-and-execute-transaction"), switchAccount: formMutationKeyFn("switch-account"), reportTransactionEffects: formMutationKeyFn("report-transaction-effects") }; function formMutationKeyFn(baseEntity) { return function mutationKeyFn(additionalKeys = []) { return [{ ...walletMutationKeys.all, baseEntity }, ...additionalKeys]; }; } // src/hooks/wallet/useWalletStore.ts import { useContext } from "react"; import { useStore } from "zustand"; function useWalletStore(selector) { const store = useContext(WalletContext); if (!store) { throw new Error( "Could not find WalletContext. Ensure that you have set up the WalletProvider." ); } return useStore(store, selector); } // src/hooks/wallet/useConnectWallet.ts function useConnectWallet({ mutationKey, ...mutationOptions } = {}) { const setWalletConnected = useWalletStore((state) => state.setWalletConnected); const setConnectionStatus = useWalletStore((state) => state.setConnectionStatus); return useMutation({ mutationKey: walletMutationKeys.connectWallet(mutationKey), mutationFn: async ({ wallet, accountAddress, ...connectArgs }) => { try { setConnectionStatus("connecting"); const connectResult = await wallet.features["standard:connect"].connect(connectArgs); const connectedSuiAccounts = connectResult.accounts.filter( (account) => account.chains.some((chain) => chain.split(":")[0] === "sui") ); const selectedAccount = getSelectedAccount(connectedSuiAccounts, accountAddress); setWalletConnected( wallet, connectedSuiAccounts, selectedAccount, connectResult.supportedIntents ); return { accounts: connectedSuiAccounts }; } catch (error) { setConnectionStatus("disconnected"); throw error; } }, ...mutationOptions }); } function getSelectedAccount(connectedAccounts, accountAddress) { if (connectedAccounts.length === 0) { return null; } if (accountAddress) { const selectedAccount = connectedAccounts.find((account) => account.address === accountAddress); return selectedAccount ?? connectedAccounts[0]; } return connectedAccounts[0]; } // src/hooks/wallet/useCurrentWallet.ts function useCurrentWallet() { const currentWallet = useWalletStore((state) => state.currentWallet); const connectionStatus = useWalletStore((state) => state.connectionStatus); const supportedIntents = useWalletStore((state) => state.supportedIntents); switch (connectionStatus) { case "connecting": return { connectionStatus, currentWallet: null, isDisconnected: false, isConnecting: true, isConnected: false, supportedIntents: [] }; case "disconnected": return { connectionStatus, currentWallet: null, isDisconnected: true, isConnecting: false, isConnected: false, supportedIntents: [] }; case "connected": { return { connectionStatus, currentWallet, isDisconnected: false, isConnecting: false, isConnected: true, supportedIntents }; } } } // src/hooks/wallet/useWallets.ts function useWallets() { return useWalletStore((state) => state.wallets); } // src/hooks/wallet/useAutoConnectWallet.ts function useAutoConnectWallet() { const { mutateAsync: connectWallet } = useConnectWallet(); const autoConnectEnabled = useWalletStore((state) => state.autoConnectEnabled); const lastConnectedWalletName = useWalletStore((state) => state.lastConnectedWalletName); const lastConnectedAccountAddress = useWalletStore((state) => state.lastConnectedAccountAddress); const wallets = useWallets(); const { isConnected } = useCurrentWallet(); const [clientOnly, setClientOnly] = useState(false); useLayoutEffect(() => { setClientOnly(true); }, []); const { data, isError } = useQuery({ queryKey: [ "@mysten/dapp-kit", "autoconnect", { isConnected, autoConnectEnabled, lastConnectedWalletName, lastConnectedAccountAddress, walletCount: wallets.length } ], queryFn: async () => { if (!autoConnectEnabled) { return "disabled"; } if (!lastConnectedWalletName || !lastConnectedAccountAddress || isConnected) { return "attempted"; } const wallet = wallets.find( (wallet2) => getWalletUniqueIdentifier(wallet2) === lastConnectedWalletName ); if (wallet) { await connectWallet({ wallet, accountAddress: lastConnectedAccountAddress, silent: true }); } return "attempted"; }, enabled: autoConnectEnabled, persister: void 0, gcTime: 0, staleTime: 0, networkMode: "always", retry: false, retryOnMount: false, refetchInterval: false, refetchIntervalInBackground: false, refetchOnMount: false, refetchOnReconnect: false, refetchOnWindowFocus: false }); if (!autoConnectEnabled) { return "disabled"; } if (!clientOnly) { return "idle"; } if (isConnected) { return "attempted"; } if (!lastConnectedWalletName) { return "attempted"; } return isError ? "attempted" : data ?? "idle"; } // src/hooks/wallet/useSlushWallet.ts import { registerSlushWallet } from "@mysten/slush-wallet"; import { useLayoutEffect as useLayoutEffect2 } from "react"; function useSlushWallet(config) { useLayoutEffect2(() => { if (!config?.name) { return; } let cleanup; let isMounted = true; try { const result = registerSlushWallet(config.name, { origin: config.origin }); if (isMounted && result) { cleanup = result.unregister; } else if (result) { result.unregister(); } } catch (error) { console.error("Failed to register Slush wallet:", error); } return () => { isMounted = false; if (cleanup) cleanup(); }; }, [config?.name, config?.origin]); } // src/hooks/wallet/useUnsafeBurnerWallet.ts import { Ed25519Keypair } from "@mysten/sui/keypairs/ed25519"; import { Transaction } from "@mysten/sui/transactions"; import { toBase64 } from "@mysten/sui/utils"; import { getWallets as getWallets2, ReadonlyWalletAccount, SUI_CHAINS } from "@mysten/wallet-standard"; import { useEffect } from "react"; // src/hooks/useSuiClient.ts import { useContext as useContext2 } from "react"; // src/components/SuiClientProvider.tsx import { getFullnodeUrl, isSuiClient, SuiClient } from "@mysten/sui/client"; import { createContext as createContext2, useMemo, useState as useState2 } from "react"; import { jsx } from "react/jsx-runtime"; var SuiClientContext = createContext2(null); var DEFAULT_NETWORKS = { localnet: { url: getFullnodeUrl("localnet") } }; // src/hooks/useSuiClient.ts function useSuiClientContext() { const suiClient = useContext2(SuiClientContext); if (!suiClient) { throw new Error( "Could not find SuiClientContext. Ensure that you have set up the SuiClientProvider" ); } return suiClient; } function useSuiClient() { return useSuiClientContext().client; } // src/hooks/wallet/useUnsafeBurnerWallet.ts var WALLET_NAME = "Unsafe Burner Wallet"; function useUnsafeBurnerWallet(enabled) { const suiClient = useSuiClient(); useEffect(() => { if (!enabled) { return; } const unregister = registerUnsafeBurnerWallet(suiClient); return unregister; }, [enabled, suiClient]); } function registerUnsafeBurnerWallet(suiClient) { var _on, _connect, _signPersonalMessage, _signTransactionBlock, _signTransaction, _signAndExecuteTransactionBlock, _signAndExecuteTransaction; const walletsApi = getWallets2(); const registeredWallets = walletsApi.get(); if (registeredWallets.find((wallet) => wallet.name === WALLET_NAME)) { console.warn( "registerUnsafeBurnerWallet: Unsafe Burner Wallet already registered, skipping duplicate registration." ); return; } console.warn( "Your application is currently using the unsafe burner wallet. Make sure that this wallet is disabled in production." ); const keypair = new Ed25519Keypair(); const account = new ReadonlyWalletAccount({ address: keypair.getPublicKey().toSuiAddress(), publicKey: keypair.getPublicKey().toSuiBytes(), chains: ["sui:unknown"], features: [ "sui:signAndExecuteTransactionBlock", "sui:signTransactionBlock", "sui:signTransaction", "sui:signAndExecuteTransaction" ] }); class UnsafeBurnerWallet { constructor() { __privateAdd(this, _on, () => { return () => { }; }); __privateAdd(this, _connect, async () => { return { accounts: this.accounts }; }); __privateAdd(this, _signPersonalMessage, async (messageInput) => { const { bytes, signature } = await keypair.signPersonalMessage(messageInput.message); return { bytes, signature }; }); __privateAdd(this, _signTransactionBlock, async (transactionInput) => { const { bytes, signature } = await transactionInput.transactionBlock.sign({ client: suiClient, signer: keypair }); return { transactionBlockBytes: bytes, signature }; }); __privateAdd(this, _signTransaction, async (transactionInput) => { const { bytes, signature } = await Transaction.from( await transactionInput.transaction.toJSON() ).sign({ client: suiClient, signer: keypair }); transactionInput.signal?.throwIfAborted(); return { bytes, signature }; }); __privateAdd(this, _signAndExecuteTransactionBlock, async (transactionInput) => { const { bytes, signature } = await transactionInput.transactionBlock.sign({ client: suiClient, signer: keypair }); return suiClient.executeTransactionBlock({ signature, transactionBlock: bytes, options: transactionInput.options }); }); __privateAdd(this, _signAndExecuteTransaction, async (transactionInput) => { const { bytes, signature } = await Transaction.from( await transactionInput.transaction.toJSON() ).sign({ client: suiClient, signer: keypair }); transactionInput.signal?.throwIfAborted(); const { rawEffects, digest } = await suiClient.executeTransactionBlock({ signature, transactionBlock: bytes, options: { showRawEffects: true } }); return { bytes, signature, digest, effects: toBase64(new Uint8Array(rawEffects)) }; }); } get version() { return "1.0.0"; } get name() { return WALLET_NAME; } get icon() { return ""; } // Return the Sui chains that your wallet supports. get chains() { return SUI_CHAINS; } get accounts() { return [account]; } get features() { return { "standard:connect": { version: "1.0.0", connect: __privateGet(this, _connect) }, "standard:events": { version: "1.0.0", on: __privateGet(this, _on) }, "sui:signPersonalMessage": { version: "1.1.0", signPersonalMessage: __privateGet(this, _signPersonalMessage) }, "sui:signTransactionBlock": { version: "1.0.0", signTransactionBlock: __privateGet(this, _signTransactionBlock) }, "sui:signAndExecuteTransactionBlock": { version: "1.0.0", signAndExecuteTransactionBlock: __privateGet(this, _signAndExecuteTransactionBlock) }, "sui:signTransaction": { version: "2.0.0", signTransaction: __privateGet(this, _signTransaction) }, "sui:signAndExecuteTransaction": { version: "2.0.0", signAndExecuteTransaction: __privateGet(this, _signAndExecuteTransaction) } }; } } _on = new WeakMap(); _connect = new WeakMap(); _signPersonalMessage = new WeakMap(); _signTransactionBlock = new WeakMap(); _signTransaction = new WeakMap(); _signAndExecuteTransactionBlock = new WeakMap(); _signAndExecuteTransaction = new WeakMap(); return walletsApi.register(new UnsafeBurnerWallet()); } // src/hooks/wallet/useWalletPropertiesChanged.ts import { useEffect as useEffect2 } from "react"; function useWalletPropertiesChanged() { const { currentWallet } = useCurrentWallet(); const updateWalletAccounts = useWalletStore((state) => state.updateWalletAccounts); useEffect2(() => { const unsubscribeFromEvents = currentWallet?.features["standard:events"].on( "change", ({ accounts }) => { if (accounts) { updateWalletAccounts(accounts); } } ); return unsubscribeFromEvents; }, [currentWallet?.features, updateWalletAccounts]); } // src/hooks/wallet/useWalletsChanged.ts import { getWallets as getWallets3 } from "@mysten/wallet-standard"; import { useEffect as useEffect3 } from "react"; function useWalletsChanged(preferredWallets, walletFilter) { const setWalletRegistered = useWalletStore((state) => state.setWalletRegistered); const setWalletUnregistered = useWalletStore((state) => state.setWalletUnregistered); useEffect3(() => { const walletsApi = getWallets3(); setWalletRegistered(getRegisteredWallets(preferredWallets, walletFilter)); const unsubscribeFromRegister = walletsApi.on("register", () => { setWalletRegistered(getRegisteredWallets(preferredWallets, walletFilter)); }); const unsubscribeFromUnregister = walletsApi.on("unregister", (unregisteredWallet) => { setWalletUnregistered( getRegisteredWallets(preferredWallets, walletFilter), unregisteredWallet ); }); return () => { unsubscribeFromRegister(); unsubscribeFromUnregister(); }; }, [preferredWallets, walletFilter, setWalletRegistered, setWalletUnregistered]); } // src/themes/lightTheme.ts var lightTheme = { blurs: { modalOverlay: "blur(0)" }, backgroundColors: { primaryButton: "#F6F7F9", primaryButtonHover: "#F0F2F5", outlineButtonHover: "#F4F4F5", modalOverlay: "rgba(24 36 53 / 20%)", modalPrimary: "white", modalSecondary: "#F7F8F8", iconButton: "transparent", iconButtonHover: "#F0F1F2", dropdownMenu: "#FFFFFF", dropdownMenuSeparator: "#F3F6F8", walletItemSelected: "white", walletItemHover: "#3C424226" }, borderColors: { outlineButton: "#E4E4E7" }, colors: { primaryButton: "#373737", outlineButton: "#373737", iconButton: "#000000", body: "#182435", bodyMuted: "#767A81", bodyDanger: "#FF794B" }, radii: { small: "6px", medium: "8px", large: "12px", xlarge: "16px" }, shadows: { primaryButton: "0px 4px 12px rgba(0, 0, 0, 0.1)", walletItemSelected: "0px 2px 6px rgba(0, 0, 0, 0.05)" }, fontWeights: { normal: "400", medium: "500", bold: "600" }, fontSizes: { small: "14px", medium: "16px", large: "18px", xlarge: "20px" }, typography: { fontFamily: 'ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"', fontStyle: "normal", lineHeight: "1.3", letterSpacing: "1" } }; // src/walletStore.ts import { createStore } from "zustand"; import { createJSONStorage, persist } from "zustand/middleware"; function createWalletStore({ wallets, storage, storageKey, autoConnectEnabled }) { return createStore()( persist( (set, get) => ({ autoConnectEnabled, wallets, accounts: [], currentWallet: null, currentAccount: null, lastConnectedAccountAddress: null, lastConnectedWalletName: null, connectionStatus: "disconnected", supportedIntents: [], setConnectionStatus(connectionStatus) { set(() => ({ connectionStatus })); }, setWalletConnected(wallet, connectedAccounts, selectedAccount, supportedIntents = []) { set(() => ({ accounts: connectedAccounts, currentWallet: wallet, currentAccount: selectedAccount, lastConnectedWalletName: getWalletUniqueIdentifier(wallet), lastConnectedAccountAddress: selectedAccount?.address, connectionStatus: "connected", supportedIntents })); }, setWalletDisconnected() { set(() => ({ accounts: [], currentWallet: null, currentAccount: null, lastConnectedWalletName: null, lastConnectedAccountAddress: null, connectionStatus: "disconnected", supportedIntents: [] })); }, setAccountSwitched(selectedAccount) { set(() => ({ currentAccount: selectedAccount, lastConnectedAccountAddress: selectedAccount.address })); }, setWalletRegistered(updatedWallets) { set(() => ({ wallets: updatedWallets })); }, setWalletUnregistered(updatedWallets, unregisteredWallet) { if (unregisteredWallet === get().currentWallet) { set(() => ({ wallets: updatedWallets, accounts: [], currentWallet: null, currentAccount: null, lastConnectedWalletName: null, lastConnectedAccountAddress: null, connectionStatus: "disconnected", supportedIntents: [] })); } else { set(() => ({ wallets: updatedWallets })); } }, updateWalletAccounts(accounts) { const currentAccount = get().currentAccount; set(() => ({ accounts, currentAccount: currentAccount && accounts.find(({ address }) => address === currentAccount.address) || accounts[0] })); } }), { name: storageKey, storage: createJSONStorage(() => storage), partialize: ({ lastConnectedWalletName, lastConnectedAccountAddress }) => ({ lastConnectedWalletName, lastConnectedAccountAddress }) } ) ); } // src/components/styling/InjectedThemeStyles.tsx import { assignInlineVars } from "@vanilla-extract/dynamic"; // src/constants/styleDataAttribute.ts var styleDataAttributeName = "data-dapp-kit"; var styleDataAttributeSelector = `[${styleDataAttributeName}]`; var styleDataAttribute = { [styleDataAttributeName]: "" }; // src/themes/themeContract.ts import { createGlobalThemeContract } from "@vanilla-extract/css"; var themeContractValues = { blurs: { modalOverlay: "" }, backgroundColors: { primaryButton: "", primaryButtonHover: "", outlineButtonHover: "", walletItemHover: "", walletItemSelected: "", modalOverlay: "", modalPrimary: "", modalSecondary: "", iconButton: "", iconButtonHover: "", dropdownMenu: "", dropdownMenuSeparator: "" }, borderColors: { outlineButton: "" }, colors: { primaryButton: "", outlineButton: "", body: "", bodyMuted: "", bodyDanger: "", iconButton: "" }, radii: { small: "", medium: "", large: "", xlarge: "" }, shadows: { primaryButton: "", walletItemSelected: "" }, fontWeights: { normal: "", medium: "", bold: "" }, fontSizes: { small: "", medium: "", large: "", xlarge: "" }, typography: { fontFamily: "", fontStyle: "", lineHeight: "", letterSpacing: "" } }; var themeVars = createGlobalThemeContract( themeContractValues, (_, path) => `dapp-kit-${path.join("-")}` ); // src/components/styling/InjectedThemeStyles.tsx import { jsx as jsx2 } from "react/jsx-runtime"; function InjectedThemeStyles({ theme }) { const themeStyles = Array.isArray(theme) ? getDynamicThemeStyles(theme) : getStaticThemeStyles(theme); return /* @__PURE__ */ jsx2( "style", { precedence: "default", href: "mysten-dapp-kit-theme", dangerouslySetInnerHTML: { __html: themeStyles } } ); } function getDynamicThemeStyles(themes) { return themes.map(({ mediaQuery, selector, variables }) => { const themeStyles = getStaticThemeStyles(variables); const themeStylesWithSelectorPrefix = selector ? `${selector} ${themeStyles}` : themeStyles; return mediaQuery ? `@media ${mediaQuery}{${themeStylesWithSelectorPrefix}}` : themeStylesWithSelectorPrefix; }).join(" "); } function getStaticThemeStyles(theme) { return `${styleDataAttributeSelector} {${cssStringFromTheme(theme)}}`; } function cssStringFromTheme(theme) { return Object.entries(assignInlineVars(themeVars, theme)).map(([key, value]) => `${key}:${value};`).join(""); } // src/components/WalletProvider.tsx import { jsx as jsx3, jsxs } from "react/jsx-runtime"; function WalletProvider({ preferredWallets = DEFAULT_PREFERRED_WALLETS, walletFilter = DEFAULT_WALLET_FILTER, storage = DEFAULT_STORAGE, storageKey = DEFAULT_STORAGE_KEY, enableUnsafeBurner = false, autoConnect = false, slushWallet, theme = lightTheme, children }) { const storeRef = useRef( createWalletStore({ autoConnectEnabled: autoConnect, wallets: getRegisteredWallets(preferredWallets, walletFilter), storage: storage || createInMemoryStore(), storageKey }) ); return /* @__PURE__ */ jsx3(WalletContext.Provider, { value: storeRef.current, children: /* @__PURE__ */ jsxs( WalletConnectionManager, { preferredWallets, walletFilter, enableUnsafeBurner, slushWallet, children: [ theme ? /* @__PURE__ */ jsx3(InjectedThemeStyles, { theme }) : null, children ] } ) }); } function WalletConnectionManager({ preferredWallets = DEFAULT_PREFERRED_WALLETS, walletFilter = DEFAULT_WALLET_FILTER, enableUnsafeBurner = false, slushWallet, children }) { useWalletsChanged(preferredWallets, walletFilter); useWalletPropertiesChanged(); useSlushWallet(slushWallet); useUnsafeBurnerWallet(enableUnsafeBurner); useAutoConnectWallet(); return children; } export { WalletProvider }; //# sourceMappingURL=WalletProvider.js.map