import React, { useEffect, useState } from "react"; import { version } from "@walletconnect/client/package.json"; import * as encoding from "@walletconnect/encoding"; 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"; import { DEFAULT_MAIN_CHAINS, DEFAULT_TEST_CHAINS } from "./constants"; import { getAllChainNamespaces, AccountAction, ChainNamespaces, getLocalStorageTestnetFlag, setLocaleStorageTestnetFlag, } from "./helpers"; import Toggle from "./components/Toggle"; import RequestModal from "./modals/RequestModal"; import PairingModal from "./modals/PairingModal"; import PingModal from "./modals/PingModal"; import { SAccounts, SAccountsContainer, SButtonContainer, SConnectButton, SContent, SLanding, SLayout, SToggleContainer, } from "./components/app"; import { useWalletConnectClient } from "./contexts/ClientContext"; import { apiGetChainNamespace, ChainsMap } from "caip-api"; import { utils } from "ethers"; interface IFormattedRpcResponse { method: string; address: string; valid: boolean; result: string; } export default function App() { const [isTestnet, setIsTestnet] = useState(getLocalStorageTestnetFlag()); const [chainData, setChainData] = useState({}); const [isRpcRequestPending, setIsRpcRequestPending] = useState(false); const [rpcResult, setRpcResult] = useState(); const [modal, setModal] = useState(""); const closeModal = () => setModal(""); const openPairingModal = () => setModal("pairing"); const openPingModal = () => setModal("ping"); const openRequestModal = () => setModal("request"); // Initialize the WalletConnect client. const { client, session, disconnect, chain, accounts, balances, isFetchingBalances, isInitializing, onEnable, web3Provider, } = useWalletConnectClient(); // Close the pairing modal after a session is established. useEffect(() => { if (session && modal === "pairing") { closeModal(); } }, [session, modal]); useEffect(() => { loadChainData(); }, []); const loadChainData = async () => { const namespaces = getAllChainNamespaces(); const chainData: ChainNamespaces = {}; await Promise.all( namespaces.map(async namespace => { let chains: ChainsMap | undefined; try { chains = await apiGetChainNamespace(namespace); } catch (e) { // ignore error } if (typeof chains !== "undefined") { chainData[namespace] = chains; } }), ); setChainData(chainData); }; // TODO: // const onPing = async () => { // openPingModal(); // await ping(); // }; const testSignMessage: () => Promise = async () => { if (!web3Provider) { throw new Error("web3Provider not connected"); } const msg = "hello world"; const hexMsg = encoding.utf8ToHex(msg, true); const address = accounts[0]; const signature = await web3Provider.send("personal_sign", [hexMsg, address]); const valid = utils.verifyMessage(msg, signature) === address; return { method: "personal_sign", address, valid, result: signature, }; }; const getEthereumActions = (): AccountAction[] => { const onSignPersonalMessage = async () => { openRequestModal(); try { setIsRpcRequestPending(true); const result = await testSignMessage(); setRpcResult(result); } catch (error) { console.error(error); } finally { setIsRpcRequestPending(false); } }; return [ // { method: "eth_sendTransaction", callback: onSendTransaction }, { method: "personal_sign", callback: onSignPersonalMessage }, // { method: "eth_signTypedData", callback: onSignTypedData }, ]; }; const getBlockchainActions = (chainId: string) => { const [namespace] = chainId.split(":"); switch (namespace) { case "eip155": return getEthereumActions(); case "cosmos": return []; default: break; } }; // Toggle between displaying testnet or mainnet chains as selection options. const toggleTestnets = () => { const nextIsTestnetState = !isTestnet; setIsTestnet(nextIsTestnetState); setLocaleStorageTestnetFlag(nextIsTestnetState); }; // Renders the appropriate model for the given request that is currently in-flight. const renderModal = () => { switch (modal) { case "pairing": if (typeof client === "undefined") { throw new Error("WalletConnect is not initialized"); } // return ; return null; case "request": return ; case "ping": return ; default: return null; } }; const renderContent = () => { const chainOptions = isTestnet ? DEFAULT_TEST_CHAINS : DEFAULT_MAIN_CHAINS; return !accounts.length && !Object.keys(balances).length ? (
{`Using v${version || "2.0.0-beta"}`}
Select an Ethereum chain:

Testnets Only?

{chainOptions.map(chainId => ( ))} {/* {"Connect"} */}
) : (

Account

{accounts.map(account => { return ( ); })}
); }; return (
Promise.resolve()} disconnect={disconnect} session={session} /> {isInitializing ? "Loading..." : renderContent()} {renderModal()} ); }