2022-02-10 10:45:19 +00:00
|
|
|
import React, { useEffect, useState } from "react";
|
2022-05-30 09:46:15 +00:00
|
|
|
import { version } from "@walletconnect/sign-client/package.json";
|
2022-02-02 13:00:33 +00:00
|
|
|
|
|
|
|
import Banner from "./components/Banner";
|
|
|
|
import Blockchain from "./components/Blockchain";
|
|
|
|
import Column from "./components/Column";
|
|
|
|
import Header from "./components/Header";
|
|
|
|
import Modal from "./components/Modal";
|
2022-03-17 15:21:47 +00:00
|
|
|
import {
|
|
|
|
DEFAULT_COSMOS_METHODS,
|
|
|
|
DEFAULT_EIP155_METHODS,
|
|
|
|
DEFAULT_MAIN_CHAINS,
|
|
|
|
DEFAULT_SOLANA_METHODS,
|
|
|
|
DEFAULT_TEST_CHAINS,
|
|
|
|
} from "./constants";
|
2022-03-16 17:00:20 +00:00
|
|
|
import { AccountAction, setLocaleStorageTestnetFlag } from "./helpers";
|
2022-02-02 13:00:33 +00:00
|
|
|
import Toggle from "./components/Toggle";
|
|
|
|
import RequestModal from "./modals/RequestModal";
|
|
|
|
import PairingModal from "./modals/PairingModal";
|
|
|
|
import PingModal from "./modals/PingModal";
|
2022-02-10 10:45:19 +00:00
|
|
|
import {
|
|
|
|
SAccounts,
|
|
|
|
SAccountsContainer,
|
|
|
|
SButtonContainer,
|
|
|
|
SConnectButton,
|
|
|
|
SContent,
|
|
|
|
SLanding,
|
|
|
|
SLayout,
|
|
|
|
SToggleContainer,
|
|
|
|
} from "./components/app";
|
|
|
|
import { useWalletConnectClient } from "./contexts/ClientContext";
|
|
|
|
import { useJsonRpc } from "./contexts/JsonRpcContext";
|
2022-03-17 15:56:54 +00:00
|
|
|
import { useChainData } from "./contexts/ChainDataContext";
|
2022-02-10 10:45:19 +00:00
|
|
|
|
|
|
|
export default function App() {
|
|
|
|
const [modal, setModal] = useState("");
|
|
|
|
|
|
|
|
const closeModal = () => setModal("");
|
|
|
|
const openPairingModal = () => setModal("pairing");
|
|
|
|
const openPingModal = () => setModal("ping");
|
|
|
|
const openRequestModal = () => setModal("request");
|
|
|
|
|
2022-02-10 15:53:46 +00:00
|
|
|
// Initialize the WalletConnect client.
|
2022-02-10 10:45:19 +00:00
|
|
|
const {
|
|
|
|
client,
|
|
|
|
session,
|
|
|
|
connect,
|
|
|
|
disconnect,
|
|
|
|
chains,
|
|
|
|
accounts,
|
|
|
|
balances,
|
2022-02-10 16:16:08 +00:00
|
|
|
isFetchingBalances,
|
2022-02-10 16:12:38 +00:00
|
|
|
isInitializing,
|
2022-02-10 10:45:19 +00:00
|
|
|
setChains,
|
|
|
|
} = useWalletConnectClient();
|
|
|
|
|
2022-02-10 15:53:46 +00:00
|
|
|
// Use `JsonRpcContext` to provide us with relevant RPC methods and states.
|
2022-03-16 17:00:20 +00:00
|
|
|
const {
|
|
|
|
ping,
|
|
|
|
ethereumRpc,
|
|
|
|
cosmosRpc,
|
|
|
|
solanaRpc,
|
|
|
|
isRpcRequestPending,
|
|
|
|
rpcResult,
|
|
|
|
isTestnet,
|
|
|
|
setIsTestnet,
|
|
|
|
} = useJsonRpc();
|
2022-02-10 10:45:19 +00:00
|
|
|
|
2022-03-17 15:56:54 +00:00
|
|
|
const { chainData } = useChainData();
|
|
|
|
|
2022-02-10 15:53:46 +00:00
|
|
|
// Close the pairing modal after a session is established.
|
2022-02-10 10:45:19 +00:00
|
|
|
useEffect(() => {
|
|
|
|
if (session && modal === "pairing") {
|
|
|
|
closeModal();
|
|
|
|
}
|
|
|
|
}, [session, modal]);
|
|
|
|
|
|
|
|
const onConnect = () => {
|
|
|
|
if (typeof client === "undefined") {
|
2022-02-02 13:00:33 +00:00
|
|
|
throw new Error("WalletConnect is not initialized");
|
|
|
|
}
|
2022-02-10 15:53:46 +00:00
|
|
|
// Suggest existing pairings (if any).
|
2022-05-30 09:46:15 +00:00
|
|
|
if (client.pairing.values.length) {
|
|
|
|
openPairingModal();
|
|
|
|
} else {
|
|
|
|
// If no existing pairings are available, trigger `WalletConnectClient.connect`.
|
|
|
|
connect();
|
2022-02-02 13:00:33 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
const onPing = async () => {
|
|
|
|
openPingModal();
|
|
|
|
await ping();
|
2022-02-02 13:00:33 +00:00
|
|
|
};
|
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
const getEthereumActions = (): AccountAction[] => {
|
2022-02-21 10:31:07 +00:00
|
|
|
const onSendTransaction = async (chainId: string, address: string) => {
|
2022-02-10 10:45:19 +00:00
|
|
|
openRequestModal();
|
2022-02-21 10:31:07 +00:00
|
|
|
await ethereumRpc.testSendTransaction(chainId, address);
|
2022-02-10 10:45:19 +00:00
|
|
|
};
|
2022-02-21 10:31:07 +00:00
|
|
|
const onSignTransaction = async (chainId: string, address: string) => {
|
2022-02-16 13:05:12 +00:00
|
|
|
openRequestModal();
|
2022-02-21 10:31:07 +00:00
|
|
|
await ethereumRpc.testSignTransaction(chainId, address);
|
2022-02-16 13:05:12 +00:00
|
|
|
};
|
2022-02-21 10:31:07 +00:00
|
|
|
const onSignPersonalMessage = async (chainId: string, address: string) => {
|
2022-02-10 10:45:19 +00:00
|
|
|
openRequestModal();
|
2022-02-21 10:31:07 +00:00
|
|
|
await ethereumRpc.testSignPersonalMessage(chainId, address);
|
2022-02-10 10:45:19 +00:00
|
|
|
};
|
2022-02-21 10:31:07 +00:00
|
|
|
const onEthSign = async (chainId: string, address: string) => {
|
2022-02-16 13:05:12 +00:00
|
|
|
openRequestModal();
|
2022-02-21 10:31:07 +00:00
|
|
|
await ethereumRpc.testEthSign(chainId, address);
|
2022-02-16 13:05:12 +00:00
|
|
|
};
|
2022-02-21 10:31:07 +00:00
|
|
|
const onSignTypedData = async (chainId: string, address: string) => {
|
2022-02-10 10:45:19 +00:00
|
|
|
openRequestModal();
|
2022-02-21 10:31:07 +00:00
|
|
|
await ethereumRpc.testSignTypedData(chainId, address);
|
2022-02-10 10:45:19 +00:00
|
|
|
};
|
2022-02-02 13:00:33 +00:00
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
return [
|
2022-03-17 15:21:47 +00:00
|
|
|
{ method: DEFAULT_EIP155_METHODS.ETH_SEND_TRANSACTION, callback: onSendTransaction },
|
|
|
|
{ method: DEFAULT_EIP155_METHODS.ETH_SIGN_TRANSACTION, callback: onSignTransaction },
|
|
|
|
{ method: DEFAULT_EIP155_METHODS.PERSONAL_SIGN, callback: onSignPersonalMessage },
|
|
|
|
{ method: DEFAULT_EIP155_METHODS.ETH_SIGN + " (standard)", callback: onEthSign },
|
|
|
|
{ method: DEFAULT_EIP155_METHODS.ETH_SIGN_TYPED_DATA, callback: onSignTypedData },
|
2022-02-10 10:45:19 +00:00
|
|
|
];
|
2022-02-02 13:00:33 +00:00
|
|
|
};
|
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
const getCosmosActions = (): AccountAction[] => {
|
2022-02-21 10:31:07 +00:00
|
|
|
const onSignDirect = async (chainId: string, address: string) => {
|
2022-02-10 10:45:19 +00:00
|
|
|
openRequestModal();
|
2022-02-21 10:31:07 +00:00
|
|
|
await cosmosRpc.testSignDirect(chainId, address);
|
2022-02-10 10:45:19 +00:00
|
|
|
};
|
2022-02-21 10:31:07 +00:00
|
|
|
const onSignAmino = async (chainId: string, address: string) => {
|
2022-02-10 10:45:19 +00:00
|
|
|
openRequestModal();
|
2022-02-21 10:31:07 +00:00
|
|
|
await cosmosRpc.testSignAmino(chainId, address);
|
2022-02-10 10:45:19 +00:00
|
|
|
};
|
|
|
|
return [
|
2022-03-17 15:21:47 +00:00
|
|
|
{ method: DEFAULT_COSMOS_METHODS.COSMOS_SIGN_DIRECT, callback: onSignDirect },
|
|
|
|
{ method: DEFAULT_COSMOS_METHODS.COSMOS_SIGN_AMINO, callback: onSignAmino },
|
2022-02-10 10:45:19 +00:00
|
|
|
];
|
2022-02-02 13:00:33 +00:00
|
|
|
};
|
|
|
|
|
2022-03-16 17:00:20 +00:00
|
|
|
const getSolanaActions = (): AccountAction[] => {
|
|
|
|
const onSignTransaction = async (chainId: string, address: string) => {
|
|
|
|
openRequestModal();
|
|
|
|
await solanaRpc.testSignTransaction(chainId, address);
|
|
|
|
};
|
|
|
|
const onSignMessage = async (chainId: string, address: string) => {
|
|
|
|
openRequestModal();
|
|
|
|
await solanaRpc.testSignMessage(chainId, address);
|
|
|
|
};
|
|
|
|
return [
|
|
|
|
{ method: DEFAULT_SOLANA_METHODS.SOL_SIGN_TRANSACTION, callback: onSignTransaction },
|
|
|
|
{ method: DEFAULT_SOLANA_METHODS.SOL_SIGN_MESSAGE, callback: onSignMessage },
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
const getBlockchainActions = (chainId: string) => {
|
2022-02-02 13:00:33 +00:00
|
|
|
const [namespace] = chainId.split(":");
|
|
|
|
switch (namespace) {
|
|
|
|
case "eip155":
|
2022-02-10 10:45:19 +00:00
|
|
|
return getEthereumActions();
|
2022-02-02 13:00:33 +00:00
|
|
|
case "cosmos":
|
2022-02-10 10:45:19 +00:00
|
|
|
return getCosmosActions();
|
2022-03-16 17:00:20 +00:00
|
|
|
case "solana":
|
|
|
|
return getSolanaActions();
|
2022-02-02 13:00:33 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-02-10 16:08:22 +00:00
|
|
|
// Toggle between displaying testnet or mainnet chains as selection options.
|
2022-02-10 10:45:19 +00:00
|
|
|
const toggleTestnets = () => {
|
|
|
|
const nextIsTestnetState = !isTestnet;
|
|
|
|
setIsTestnet(nextIsTestnetState);
|
2022-02-10 15:53:46 +00:00
|
|
|
setLocaleStorageTestnetFlag(nextIsTestnetState);
|
2022-02-02 13:00:33 +00:00
|
|
|
};
|
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
const handleChainSelectionClick = (chainId: string) => {
|
|
|
|
if (chains.includes(chainId)) {
|
|
|
|
setChains(chains.filter(chain => chain !== chainId));
|
|
|
|
} else {
|
|
|
|
setChains([...chains, chainId]);
|
|
|
|
}
|
2022-02-02 13:00:33 +00:00
|
|
|
};
|
|
|
|
|
2022-02-10 16:08:22 +00:00
|
|
|
// Renders the appropriate model for the given request that is currently in-flight.
|
2022-02-10 10:45:19 +00:00
|
|
|
const renderModal = () => {
|
|
|
|
switch (modal) {
|
2022-02-02 13:00:33 +00:00
|
|
|
case "pairing":
|
2022-02-10 10:45:19 +00:00
|
|
|
if (typeof client === "undefined") {
|
2022-02-02 13:00:33 +00:00
|
|
|
throw new Error("WalletConnect is not initialized");
|
|
|
|
}
|
2022-02-10 10:45:19 +00:00
|
|
|
return <PairingModal pairings={client.pairing.values} connect={connect} />;
|
2022-02-02 13:00:33 +00:00
|
|
|
case "request":
|
2022-02-10 10:45:19 +00:00
|
|
|
return <RequestModal pending={isRpcRequestPending} result={rpcResult} />;
|
2022-02-02 13:00:33 +00:00
|
|
|
case "ping":
|
2022-02-10 10:45:19 +00:00
|
|
|
return <PingModal pending={isRpcRequestPending} result={rpcResult} />;
|
2022-02-02 13:00:33 +00:00
|
|
|
default:
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
const renderContent = () => {
|
|
|
|
const chainOptions = isTestnet ? DEFAULT_TEST_CHAINS : DEFAULT_MAIN_CHAINS;
|
2022-02-02 13:00:33 +00:00
|
|
|
return !accounts.length && !Object.keys(balances).length ? (
|
|
|
|
<SLanding center>
|
|
|
|
<Banner />
|
2022-05-30 09:46:15 +00:00
|
|
|
<h6>{`Using v${version || "2.0.0-beta"}`}</h6>
|
2022-02-02 13:00:33 +00:00
|
|
|
<SButtonContainer>
|
|
|
|
<h6>Select chains:</h6>
|
|
|
|
<SToggleContainer>
|
|
|
|
<p>Testnets Only?</p>
|
2022-02-10 10:45:19 +00:00
|
|
|
<Toggle active={isTestnet} onClick={toggleTestnets} />
|
2022-02-02 13:00:33 +00:00
|
|
|
</SToggleContainer>
|
|
|
|
{chainOptions.map(chainId => (
|
|
|
|
<Blockchain
|
|
|
|
key={chainId}
|
|
|
|
chainId={chainId}
|
|
|
|
chainData={chainData}
|
2022-02-10 10:45:19 +00:00
|
|
|
onClick={handleChainSelectionClick}
|
2022-02-02 13:00:33 +00:00
|
|
|
active={chains.includes(chainId)}
|
|
|
|
/>
|
|
|
|
))}
|
2022-02-10 16:16:08 +00:00
|
|
|
<SConnectButton left onClick={onConnect} disabled={!chains.length}>
|
2022-02-02 13:00:33 +00:00
|
|
|
{"Connect"}
|
|
|
|
</SConnectButton>
|
|
|
|
</SButtonContainer>
|
|
|
|
</SLanding>
|
|
|
|
) : (
|
|
|
|
<SAccountsContainer>
|
|
|
|
<h3>Accounts</h3>
|
|
|
|
<SAccounts>
|
2022-02-10 10:45:19 +00:00
|
|
|
{accounts.map(account => {
|
2022-02-02 13:00:33 +00:00
|
|
|
const [namespace, reference, address] = account.split(":");
|
|
|
|
const chainId = `${namespace}:${reference}`;
|
|
|
|
return (
|
|
|
|
<Blockchain
|
|
|
|
key={account}
|
|
|
|
active={true}
|
|
|
|
chainData={chainData}
|
2022-02-10 16:16:08 +00:00
|
|
|
fetching={isFetchingBalances}
|
2022-02-02 13:00:33 +00:00
|
|
|
address={address}
|
|
|
|
chainId={chainId}
|
|
|
|
balances={balances}
|
2022-02-10 10:45:19 +00:00
|
|
|
actions={getBlockchainActions(chainId)}
|
2022-02-02 13:00:33 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</SAccounts>
|
|
|
|
</SAccountsContainer>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-02-10 10:45:19 +00:00
|
|
|
return (
|
|
|
|
<SLayout>
|
|
|
|
<Column maxWidth={1000} spanHeight>
|
|
|
|
<Header ping={onPing} disconnect={disconnect} session={session} />
|
2022-02-10 16:12:38 +00:00
|
|
|
<SContent>{isInitializing ? "Loading..." : renderContent()}</SContent>
|
2022-02-10 10:45:19 +00:00
|
|
|
</Column>
|
|
|
|
<Modal show={!!modal} closeModal={closeModal}>
|
|
|
|
{renderModal()}
|
|
|
|
</Modal>
|
|
|
|
</SLayout>
|
|
|
|
);
|
2022-02-02 13:00:33 +00:00
|
|
|
}
|