diff --git a/packages/thirdweb/package.json b/packages/thirdweb/package.json
index 5c2e8f34584..d7f5fa98e64 100644
--- a/packages/thirdweb/package.json
+++ b/packages/thirdweb/package.json
@@ -137,6 +137,7 @@
"!tsconfig.build.json"
],
"dependencies": {
+ "@coinbase/wallet-sdk": "^3.7.1",
"@emotion/react": "11.11.3",
"@emotion/styled": "11.11.0",
"@noble/hashes": "1.3.3",
diff --git a/packages/thirdweb/src/react/providers/thirdweb-provider-ctx.tsx b/packages/thirdweb/src/react/providers/thirdweb-provider-ctx.tsx
index 21bb3fcc0e7..db440050791 100644
--- a/packages/thirdweb/src/react/providers/thirdweb-provider-ctx.tsx
+++ b/packages/thirdweb/src/react/providers/thirdweb-provider-ctx.tsx
@@ -6,5 +6,5 @@ import type { DAppMetaData } from "../../wallets/types.js";
export const ThirdwebProviderContext = /* @__PURE__ */ createContext<{
wallets: WalletConfig[];
client: ThirdwebClient;
- dappMetadata?: DAppMetaData;
+ dappMetadata: DAppMetaData;
} | null>(null);
diff --git a/packages/thirdweb/src/react/providers/thirdweb-provider.tsx b/packages/thirdweb/src/react/providers/thirdweb-provider.tsx
index a5f8b11c88f..6fd398e5148 100644
--- a/packages/thirdweb/src/react/providers/thirdweb-provider.tsx
+++ b/packages/thirdweb/src/react/providers/thirdweb-provider.tsx
@@ -1,5 +1,3 @@
-/* eslint-disable @typescript-eslint/no-unused-vars */
-/* eslint-disable better-tree-shaking/no-top-level-side-effects */
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
import {
AutoConnect,
@@ -23,6 +21,7 @@ import {
import { getChainIdFromChain } from "../../chain/index.js";
import type { DAppMetaData } from "../../wallets/types.js";
import { isBaseTransactionOptions } from "../../transaction/index.js";
+import { defaultDappMetadata } from "../../wallets/wallet-connect/index.js";
/**
* The ThirdwebProvider is component is a provider component that sets up the React Query client and Wallet Connection Manager.
@@ -110,7 +109,7 @@ export function ThirdwebProvider(props: ThirdwebProviderProps) {
value={{
wallets: props.wallets || defaultWallets,
client: props.client,
- dappMetadata: props.dappMetadata,
+ dappMetadata: props.dappMetadata || defaultDappMetadata,
}}
>
{props.autoConnect === false ? : }
diff --git a/packages/thirdweb/src/react/types/wallets.ts b/packages/thirdweb/src/react/types/wallets.ts
index ff928d7fc59..07bcd06b5fc 100644
--- a/packages/thirdweb/src/react/types/wallets.ts
+++ b/packages/thirdweb/src/react/types/wallets.ts
@@ -16,7 +16,7 @@ export type WalletConfig = {
create: (options: {
client: ThirdwebClient;
- dappMetadata?: DAppMetaData;
+ dappMetadata: DAppMetaData;
}) => Wallet;
/**
diff --git a/packages/thirdweb/src/react/wallets/coinbase/coinbaseConfig.tsx b/packages/thirdweb/src/react/wallets/coinbase/coinbaseConfig.tsx
index 6b82d380049..fe2e97acdee 100644
--- a/packages/thirdweb/src/react/wallets/coinbase/coinbaseConfig.tsx
+++ b/packages/thirdweb/src/react/wallets/coinbase/coinbaseConfig.tsx
@@ -2,9 +2,15 @@ import {
injectedCoinbaseProvider,
coinbaseMetadata,
coinbaseWallet,
+ coinbaseSDKWallet,
+ CoinbaseSDKWallet,
} from "../../../wallets/index.js";
-import type { WalletConfig } from "../../types/wallets.js";
+import { useTWLocale } from "../../providers/locale-provider.js";
+import type { ConnectUIProps, WalletConfig } from "../../types/wallets.js";
+import { GetStartedScreen } from "../shared/GetStartedScreen.js";
import { InjectedConnectUI } from "../shared/InjectedConnectUI.js";
+import { ScanScreen } from "../shared/ScanScreen.js";
+import { useState, useRef, useEffect } from "react";
/**
* Integrate Coinbase wallet connection into your app.
@@ -19,29 +25,133 @@ import { InjectedConnectUI } from "../shared/InjectedConnectUI.js";
* @returns WalletConfig object to be passed into `ThirdwebProvider`
*/
export const coinbaseConfig = (): WalletConfig => {
+ const isInjected = !!injectedCoinbaseProvider();
+
return {
metadata: coinbaseMetadata,
- create() {
- return coinbaseWallet();
- },
- connectUI(props) {
- return (
- {
- // TODO
- }}
- // links={{
- // extension:
- // "https://chrome.google.com/webstore/detail/coinbase-wallet-extension/hnfanknocfeofbddgcijnmhnfnkdnaad",
- // android: "https://play.google.com/store/apps/details?id=org.toshi",
- // ios: "https://apps.apple.com/us/app/coinbase-wallet-nfts-crypto/id1278383455",
- // }}
- />
- );
+ create(createOptions) {
+ if (isInjected) {
+ return coinbaseWallet();
+ } else {
+ return coinbaseSDKWallet({
+ appName: createOptions.dappMetadata.name,
+ });
+ }
},
+ connectUI: CoinbaseConnectUI,
isInstalled() {
return !!injectedCoinbaseProvider();
},
};
};
+
+const links = {
+ chrome:
+ "https://chrome.google.com/webstore/detail/coinbase-wallet-extension/hnfanknocfeofbddgcijnmhnfnkdnaad",
+ android: "https://play.google.com/store/apps/details?id=org.toshi",
+ ios: "https://apps.apple.com/us/app/coinbase-wallet-nfts-crypto/id1278383455",
+};
+
+function CoinbaseConnectUI(props: ConnectUIProps) {
+ const isInjected = !!injectedCoinbaseProvider();
+ const [screen, setScreen] = useState<"main" | "get-started">("main");
+ const walletConfig = props.walletConfig;
+ const locale = useTWLocale().wallets.injectedWallet(
+ walletConfig.metadata.name,
+ );
+
+ if (screen === "get-started") {
+ return (
+ {
+ setScreen("main");
+ }}
+ />
+ );
+ }
+
+ if (isInjected) {
+ return (
+ {
+ setScreen("get-started");
+ }}
+ />
+ );
+ }
+
+ return (
+ {
+ setScreen("get-started");
+ }}
+ />
+ );
+}
+
+function CoinbaseSDKWalletConnectUI(props: {
+ connectUIProps: ConnectUIProps;
+ onGetStarted: () => void;
+}) {
+ const { connectUIProps, onGetStarted } = props;
+ const locale = useTWLocale().wallets.injectedWallet(
+ connectUIProps.walletConfig.metadata.name,
+ );
+ const { createInstance, done, chainId } = connectUIProps;
+ const [qrCodeUri, setQrCodeUri] = useState(undefined);
+
+ const scanStarted = useRef(false);
+
+ useEffect(() => {
+ if (scanStarted.current) {
+ return;
+ }
+
+ scanStarted.current = true;
+
+ (async () => {
+ const wallet = createInstance() as CoinbaseSDKWallet;
+
+ try {
+ await wallet.connect({
+ reloadOnDisconnect: false,
+ chainId,
+ onUri(uri) {
+ if (uri) {
+ setQrCodeUri(uri);
+ } else {
+ // show error
+ }
+ },
+ headlessMode: true,
+ });
+
+ done(wallet);
+ } catch {
+ // show error
+ }
+ })();
+ }, [chainId, createInstance, done]);
+
+ return (
+
+ );
+}
diff --git a/packages/thirdweb/src/wallets/coinbase/coinbaseMetadata.ts b/packages/thirdweb/src/wallets/coinbase/coinbaseMetadata.ts
new file mode 100644
index 00000000000..aae858936fa
--- /dev/null
+++ b/packages/thirdweb/src/wallets/coinbase/coinbaseMetadata.ts
@@ -0,0 +1,8 @@
+import type { WalletMetadata } from "../types.js";
+
+export const coinbaseMetadata: WalletMetadata = {
+ id: "com.coinbase.wallet",
+ name: "Coinbase Wallet",
+ iconUrl:
+ "",
+};
diff --git a/packages/thirdweb/src/wallets/coinbase/coinbaseSDKWallet.ts b/packages/thirdweb/src/wallets/coinbase/coinbaseSDKWallet.ts
new file mode 100644
index 00000000000..ee244a1b997
--- /dev/null
+++ b/packages/thirdweb/src/wallets/coinbase/coinbaseSDKWallet.ts
@@ -0,0 +1,386 @@
+import type { Account, Wallet } from "../interfaces/wallet.js";
+import type { WalletMetadata } from "../types.js";
+import type { CoinbaseWalletProvider } from "@coinbase/wallet-sdk";
+import type { CoinbaseWalletSDK as CoinbaseWalletSDKConstructor } from "@coinbase/wallet-sdk";
+import { getChainDataForChainId } from "../../chain/index.js";
+import { normalizeChainId } from "../utils/normalizeChainId.js";
+import {
+ getAddress,
+ toHex,
+ type Hex,
+ stringToHex,
+ type SignTypedDataParameters,
+ getTypesForEIP712Domain,
+ validateTypedData,
+ isHex,
+} from "viem";
+import { getValidPublicRPCUrl } from "../utils/chains.js";
+import type { SendTransactionOption } from "../interfaces/wallet.js";
+import type { Address } from "abitype";
+import { stringify } from "../../utils/json.js";
+import type { Ethereum } from "../interfaces/ethereum.js";
+import { coinbaseMetadata } from "./coinbaseMetadata.js";
+import {
+ getSavedConnectParamsFromStorage,
+ saveConnectParamsToStorage,
+} from "../manager/storage.js";
+
+type SavedConnectParams = {
+ chainId?: number;
+};
+
+type CoinbaseWalletSDKOptions = Readonly<
+ ConstructorParameters[0]
+>;
+
+export type CoinbaseSDKWalletConnectionOptions = Omit<
+ CoinbaseWalletSDKOptions,
+ "appName"
+> & {
+ chainId?: number;
+ onUri?: (uri: string | null) => void;
+};
+
+export type CoinbaseSDKWalletOptions = {
+ appName: string;
+};
+
+/**
+ * Connect to Coinbase wallet using the Coinbase SDK which allows connecting to Coinbase Wallet extension or mobile app.
+ * @param options - The options for connecting to the Coinbase Wallet SDK.
+ * @example
+ * ```ts
+ * const wallet = coinbaseSDKWallet()
+ * ```
+ * @returns A `CoinbaseSDKWallet` instance.
+ */
+export function coinbaseSDKWallet(options: CoinbaseWalletSDKOptions) {
+ return new CoinbaseSDKWallet(options);
+}
+
+/**
+ * Connect to Coinbase wallet using the Coinbase SDK which allows connecting to Coinbase Wallet extension or mobile app.
+ */
+export class CoinbaseSDKWallet implements Wallet {
+ private options: CoinbaseSDKWalletOptions;
+ private provider: CoinbaseWalletProvider | undefined;
+ private chainId: number | undefined;
+ private account?: Account | undefined;
+ metadata: WalletMetadata;
+
+ /**
+ * Create a new CoinbaseSDKWallet instance
+ * @param options - Options for connecting to the Coinbase Wallet SDK.
+ * @example
+ * ```ts
+ * const wallet = new CoinbaseSDKWallet({
+ * appName: "My App"
+ * })
+ * ```
+ * @returns A `CoinbaseSDKWallet` instance.
+ */
+ constructor(options: CoinbaseSDKWalletOptions) {
+ this.options = options;
+ this.metadata = coinbaseMetadata;
+ }
+
+ /**
+ * Get the `chainId` that the wallet is connected to.
+ * @returns The chainId
+ * @example
+ * ```ts
+ * const chainId = wallet.getChainId();
+ * ```
+ */
+ getChainId(): number | undefined {
+ return this.chainId;
+ }
+
+ /**
+ * Get the connected `Account` from the wallet.
+ * @returns The connected account
+ * @example
+ * ```ts
+ * const account = wallet.getAccount();
+ * ```
+ */
+ getAccount(): Account | undefined {
+ return this.account;
+ }
+
+ /**
+ * Connect to the Coinbase Wallet
+ * @param options - The options for connecting to the Injected Wallet Provider.
+ * @example
+ * ```ts
+ * const account = await wallet.connect()
+ * ```
+ * @returns A Promise that resolves to connected `Account` object
+ */
+ async connect(options?: CoinbaseSDKWalletConnectionOptions) {
+ const provider = await this.initProvider({
+ ...options,
+ });
+
+ provider.on("accountsChanged", this.onAccountsChanged);
+ provider.on("chainChanged", this.onChainChanged);
+ provider.on("disconnect", this.onDisconnect);
+
+ const accounts = (await provider.request({
+ method: "eth_requestAccounts",
+ })) as string[];
+
+ if (!accounts[0]) {
+ throw new Error("No accounts found");
+ }
+
+ const address = getAddress(accounts[0]);
+
+ const connectedChainId = (await provider.request({
+ method: "eth_chainId",
+ })) as string | number;
+ // TODO check what's type of connectedChainId
+
+ // Switch to chain if provided
+ if (
+ connectedChainId &&
+ options?.chainId &&
+ Number(connectedChainId) !== options?.chainId
+ ) {
+ await this.switchChain(options.chainId);
+ }
+
+ this.chainId = normalizeChainId(connectedChainId);
+
+ if (options?.chainId) {
+ const saveParams: SavedConnectParams = {
+ chainId: options?.chainId,
+ };
+
+ saveConnectParamsToStorage(this.metadata.id, saveParams);
+ }
+
+ return this.onConnect(address);
+ }
+
+ private onConnect(address: string) {
+ const wallet = this;
+
+ const account: Account = {
+ address,
+ async sendTransaction(tx: SendTransactionOption) {
+ if (!wallet.chainId || !wallet.provider || !account.address) {
+ throw new Error("Provider not setup");
+ }
+
+ if (normalizeChainId(tx.chainId) !== wallet.chainId) {
+ await wallet.switchChain(tx.chainId);
+ }
+
+ const transactionHash = (await wallet.provider.request({
+ method: "eth_sendTransaction",
+ params: [
+ {
+ accessList: tx.accessList,
+ value: tx.value ? toHex(tx.value) : undefined,
+ gas: tx.gas ? toHex(tx.gas) : undefined,
+ from: this.address,
+ to: tx.to as Address,
+ data: tx.data,
+ },
+ ],
+ })) as Hex;
+
+ return {
+ transactionHash,
+ };
+ },
+ async signMessage({ message }) {
+ if (!wallet.provider || !account.address) {
+ throw new Error("Provider not setup");
+ }
+
+ const messageToSign = (() => {
+ if (typeof message === "string") {
+ return stringToHex(message);
+ }
+ if (message.raw instanceof Uint8Array) {
+ return toHex(message.raw);
+ }
+ return message.raw;
+ })();
+
+ return await wallet.provider.request({
+ method: "personal_sign",
+ params: [messageToSign, account.address],
+ });
+ },
+ async signTypedData(typedData) {
+ if (!wallet.provider || !account.address) {
+ throw new Error("Provider not setup");
+ }
+ const { domain, message, primaryType } =
+ typedData as unknown as SignTypedDataParameters;
+
+ const types = {
+ EIP712Domain: getTypesForEIP712Domain({ domain }),
+ ...typedData.types,
+ };
+
+ // Need to do a runtime validation check on addresses, byte ranges, integer ranges, etc
+ // as we can't statically check this with TypeScript.
+ validateTypedData({ domain, message, primaryType, types });
+
+ const stringifiedData = stringify(
+ { domain: domain ?? {}, message, primaryType, types },
+ (_, value) => (isHex(value) ? value.toLowerCase() : value),
+ );
+
+ return await wallet.provider.request({
+ method: "eth_signTypedData_v4",
+ params: [account.address, stringifiedData],
+ });
+ },
+ };
+
+ this.account = account;
+ return account;
+ }
+
+ /**
+ * Auto connect to saved Coinbase Wallet session
+ * @example
+ * ```ts
+ * await wallet.autoConnect();
+ * ```
+ * @returns A Promise that resolves to the connected `Account` object
+ */
+ async autoConnect() {
+ const savedParams: SavedConnectParams | null =
+ await getSavedConnectParamsFromStorage(this.metadata.id);
+
+ const provider = await this.initProvider({
+ chainId: savedParams?.chainId,
+ });
+
+ // connected accounts
+ const addresses = await (provider as Ethereum).request({
+ method: "eth_accounts",
+ });
+
+ const address = addresses[0];
+
+ if (!address) {
+ throw new Error("No accounts found");
+ }
+
+ const connectedChainId = (await provider.request({
+ method: "eth_chainId",
+ })) as string | number;
+ this.chainId = normalizeChainId(connectedChainId);
+
+ return this.onConnect(address);
+ }
+
+ /**
+ * Switch chain in connected wallet
+ * @param chainId - The chainId to switch to
+ * @example
+ * ```ts
+ * await wallet.switchChain(1)
+ * ```
+ */
+ async switchChain(chainId: number) {
+ const provider = this.provider;
+
+ if (!provider) {
+ throw new Error("Provider not initialized");
+ }
+
+ const chainIdHex = toHex(chainId);
+
+ try {
+ await provider.request({
+ method: "wallet_switchEthereumChain",
+ params: [{ chainId: chainIdHex }],
+ });
+ } catch (error) {
+ const chain = await getChainDataForChainId(chainId);
+
+ // Indicates chain is not added to provider
+ if ((error as any).code === 4902) {
+ // try to add the chain
+ await provider.request({
+ method: "wallet_addEthereumChain",
+ params: [
+ {
+ chainId: chainIdHex,
+ chainName: chain.name,
+ nativeCurrency: chain.nativeCurrency,
+ rpcUrls: getValidPublicRPCUrl(chain), // no client id on purpose here
+ blockExplorerUrls: chain.explorers?.map((x) => x.url) || [],
+ },
+ ],
+ });
+ }
+ }
+ }
+
+ private async initProvider(options: CoinbaseSDKWalletConnectionOptions) {
+ const { CoinbaseWalletSDK } = await import("@coinbase/wallet-sdk");
+ const client = new CoinbaseWalletSDK({
+ ...options,
+ appName: this.options.appName,
+ });
+
+ if (options.onUri) {
+ options.onUri(client.getQrUrl());
+ }
+
+ const chainId = options?.chainId || 1;
+
+ const chain = await getChainDataForChainId(chainId);
+ const jsonRpcUrl = chain?.rpc[0];
+ this.provider = client.makeWeb3Provider(jsonRpcUrl, chainId);
+ return this.provider;
+ }
+
+ private onChainChanged = (chainId: number | string) => {
+ this.chainId = normalizeChainId(chainId);
+ };
+
+ private onAccountsChanged = (accounts: string[]) => {
+ if (accounts.length === 0) {
+ this.onDisconnect();
+ } else {
+ // TODO: change account
+ }
+ };
+
+ private onDisconnect = () => {
+ const provider = this.provider;
+ if (provider) {
+ provider.removeListener("accountsChanged", this.onAccountsChanged);
+ provider.removeListener("chainChanged", this.onChainChanged);
+ provider.removeListener("disconnect", this.onDisconnect);
+ }
+
+ this.account = undefined;
+ this.chainId = undefined;
+ };
+
+ /**
+ * Disconnect from the Coinbase Wallet and clear the session
+ * @example
+ * ```ts
+ * await wallet.disconnect()
+ * ```
+ */
+ async disconnect() {
+ if (this.provider) {
+ this.provider.disconnect();
+ this.provider.close();
+ }
+ this.onDisconnect();
+ }
+}
diff --git a/packages/thirdweb/src/wallets/index.ts b/packages/thirdweb/src/wallets/index.ts
index 9ebc57fef49..d72045321c0 100644
--- a/packages/thirdweb/src/wallets/index.ts
+++ b/packages/thirdweb/src/wallets/index.ts
@@ -40,7 +40,6 @@ export {
export {
injectedCoinbaseProvider,
- coinbaseMetadata,
coinbaseWallet,
} from "./injected/wallets/coinbase.js";
@@ -85,3 +84,10 @@ export {
getStoredActiveWalletId,
getStoredConnectedWalletIds,
} from "./manager/index.js";
+
+export {
+ coinbaseSDKWallet,
+ CoinbaseSDKWallet,
+ type CoinbaseSDKWalletConnectionOptions,
+} from "./coinbase/coinbaseSDKWallet.js";
+export { coinbaseMetadata } from "./coinbase/coinbaseMetadata.js";
diff --git a/packages/thirdweb/src/wallets/injected/wallets/coinbase.ts b/packages/thirdweb/src/wallets/injected/wallets/coinbase.ts
index e6abdb3d2c8..4e399501150 100644
--- a/packages/thirdweb/src/wallets/injected/wallets/coinbase.ts
+++ b/packages/thirdweb/src/wallets/injected/wallets/coinbase.ts
@@ -1,15 +1,8 @@
-import type { WalletMetadata } from "../../types.js";
+import { coinbaseMetadata } from "../../coinbase/coinbaseMetadata.js";
import { InjectedWallet } from "../index.js";
import { injectedProvider } from "../mipdStore.js";
import type { SpecificInjectedWalletOptions } from "../types.js";
-export const coinbaseMetadata: WalletMetadata = {
- id: "com.coinbase.wallet",
- name: "Coinbase Wallet",
- iconUrl:
- "",
-};
-
/**
* Connect to Injected Coinbase Wallet Provider
* @param options - The options for connecting to the Injected Coinbase Wallet Provider.
diff --git a/packages/thirdweb/tsconfig.base.json b/packages/thirdweb/tsconfig.base.json
index e92e7f13339..2b519cac7ea 100644
--- a/packages/thirdweb/tsconfig.base.json
+++ b/packages/thirdweb/tsconfig.base.json
@@ -42,8 +42,6 @@
"skipLibCheck": true,
// jsx for "/react" portion
- "jsx": "react-jsx",
-
- "stripInternal": true
+ "jsx": "react-jsx"
}
}
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index f90ee3bb8f0..38bde011cd3 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -1402,6 +1402,9 @@ importers:
packages/thirdweb:
dependencies:
+ '@coinbase/wallet-sdk':
+ specifier: ^3.7.1
+ version: 3.7.1
'@emotion/react':
specifier: 11.11.3
version: 11.11.3(@types/react@18.2.17)(react@18.2.0)
@@ -18696,7 +18699,6 @@ packages:
dependencies:
is-hex-prefixed: 1.0.0
strip-hex-prefix: 1.0.0
- bundledDependencies: false
/event-target-shim@5.0.1:
resolution: {integrity: sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==}
@@ -21261,16 +21263,6 @@ packages:
dependencies:
has-symbols: 1.0.3
- /is-typed-array@1.1.10:
- resolution: {integrity: sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==}
- engines: {node: '>= 0.4'}
- dependencies:
- available-typed-arrays: 1.0.5
- call-bind: 1.0.5
- for-each: 0.3.3
- gopd: 1.0.1
- has-tostringtag: 1.0.0
-
/is-typed-array@1.1.12:
resolution: {integrity: sha512-Z14TF2JNG8Lss5/HMqt0//T9JeHXttXy5pH/DBU4vi98ozO2btxzq9MwYDZYnKwU8nRsz/+GVFVRDq3DkVuSPg==}
engines: {node: '>= 0.4'}
@@ -28912,8 +28904,8 @@ packages:
inherits: 2.0.4
is-arguments: 1.1.1
is-generator-function: 1.0.10
- is-typed-array: 1.1.10
- which-typed-array: 1.1.9
+ is-typed-array: 1.1.12
+ which-typed-array: 1.1.13
/utils-merge@1.0.1:
resolution: {integrity: sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==}
@@ -29996,17 +29988,6 @@ packages:
gopd: 1.0.1
has-tostringtag: 1.0.0
- /which-typed-array@1.1.9:
- resolution: {integrity: sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA==}
- engines: {node: '>= 0.4'}
- dependencies:
- available-typed-arrays: 1.0.5
- call-bind: 1.0.5
- for-each: 0.3.3
- gopd: 1.0.1
- has-tostringtag: 1.0.0
- is-typed-array: 1.1.12
-
/which@1.3.1:
resolution: {integrity: sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==}
hasBin: true