diff --git a/packages/bridge-ui/.gitignore b/packages/bridge-ui/.gitignore index 1cac5597ea2..1cb393e1821 100644 --- a/packages/bridge-ui/.gitignore +++ b/packages/bridge-ui/.gitignore @@ -22,4 +22,7 @@ dist-ssr *.njsproj *.sln *.sw? -.env \ No newline at end of file +.env + +# vite +vite.config.ts.timestamp-*.mjs \ No newline at end of file diff --git a/packages/bridge-ui/jest.config.js b/packages/bridge-ui/jest.config.js index 472d18cf5ec..deb00a64f8f 100644 --- a/packages/bridge-ui/jest.config.js +++ b/packages/bridge-ui/jest.config.js @@ -39,8 +39,8 @@ export default { ], coverageThreshold: { global: { - statements: 98.45, - branches: 86, + statements: 98.36, + branches: 85, functions: 96, lines: 100, }, diff --git a/packages/bridge-ui/package.json b/packages/bridge-ui/package.json index 1702ba9902b..a76b1abf728 100644 --- a/packages/bridge-ui/package.json +++ b/packages/bridge-ui/package.json @@ -56,11 +56,14 @@ "vite-plugin-static-copy": "^0.12.0" }, "dependencies": { + "@coinbase/wallet-sdk": "^3.6.3", "@ethersproject/experimental": "^5.7.0", "@lottiefiles/svelte-lottie-player": "^0.2.0", "@sveltestack/svelte-query": "^1.6.0", - "@wagmi/core": "^0.7.5", + "@wagmi/connectors": "^0.1.1", + "@wagmi/core": "^0.8.0", "axios": "^1.2.0", + "buffer": "^6.0.3", "ethers": "^5.7.1", "extend-expect": "link:@testing-library/jest-dom/extend-expect", "identicon.js": "^2.3.3", diff --git a/packages/bridge-ui/src/App.svelte b/packages/bridge-ui/src/App.svelte index 20a477aab5b..654d6d0a88c 100644 --- a/packages/bridge-ui/src/App.svelte +++ b/packages/bridge-ui/src/App.svelte @@ -4,6 +4,15 @@ import Router from "svelte-spa-router"; import { SvelteToast } from "@zerodevx/svelte-toast"; import type { SvelteToastOptions } from "@zerodevx/svelte-toast"; + import { + configureChains, + createClient, + InjectedConnector, + } from "@wagmi/core"; + import { publicProvider } from "@wagmi/core/providers/public"; + import { jsonRpcProvider } from "@wagmi/core/providers/jsonRpc"; + import { CoinbaseWalletConnector } from "@wagmi/core/connectors/coinbaseWallet"; + import { WalletConnectConnector } from "@wagmi/core/connectors/walletConnect"; import Home from "./pages/home/Home.svelte"; import { setupI18n } from "./i18n"; @@ -19,9 +28,16 @@ import Navbar from "./components/Navbar.svelte"; import { signer } from "./store/signer"; import type { Transactioner } from "./domain/transactions"; + import { wagmiClient } from "./store/wagmi"; setupI18n({ withLocale: "en" }); - import { chains, CHAIN_MAINNET, CHAIN_TKO } from "./domain/chain"; + import { + chains, + CHAIN_MAINNET, + CHAIN_TKO, + mainnet, + taiko, + } from "./domain/chain"; import SwitchEthereumChainModal from "./components/modals/SwitchEthereumChainModal.svelte"; import { ProofService } from "./proof/service"; import { ethers } from "ethers"; @@ -45,6 +61,44 @@ CHAIN_TKO.id, new ethers.providers.JsonRpcProvider(import.meta.env.VITE_L2_RPC_URL) ); + providers.set(providerMap); + + const { + chains: wagmiChains, + provider, + webSocketProvider, + } = configureChains( + [mainnet, taiko], + [ + publicProvider(), + jsonRpcProvider({ + rpc: (chain) => ({ + http: providerMap.get(chain.id).connection.url, + }), + }), + ] + ); + + $wagmiClient = createClient({ + provider, + connectors: [ + new InjectedConnector({ + chains: wagmiChains, + }), + new CoinbaseWalletConnector({ + chains: wagmiChains, + options: { + appName: "Taiko Bridge", + }, + }), + new WalletConnectConnector({ + chains: wagmiChains, + options: { + qrcode: true, + }, + }), + ], + }); providers.set(providerMap); diff --git a/packages/bridge-ui/src/components/ChainDropdown.svelte b/packages/bridge-ui/src/components/ChainDropdown.svelte index f9eb01e7a1b..3388ef2128f 100644 --- a/packages/bridge-ui/src/components/ChainDropdown.svelte +++ b/packages/bridge-ui/src/components/ChainDropdown.svelte @@ -4,14 +4,15 @@ import ChevDown from "./icons/ChevDown.svelte"; import { fromChain, toChain } from "../store/chain"; import MetaMask from "./icons/MetaMask.svelte"; - import { switchEthereumChain } from "../utils/switchEthereumChain"; - import { ethereum } from "../store/ethereum"; import { CHAIN_MAINNET, CHAIN_TKO } from "../domain/chain"; import type { Chain } from "../domain/chain"; import { ethers } from "ethers"; import { signer } from "../store/signer"; + import { switchNetwork } from "@wagmi/core"; const changeChain = async (chain: Chain) => { - await switchEthereumChain($ethereum, chain); + await switchNetwork({ + chainId: chain.id, + }); const provider = new ethers.providers.Web3Provider(window.ethereum); await provider.send("eth_requestAccounts", []); diff --git a/packages/bridge-ui/src/components/Transaction.svelte b/packages/bridge-ui/src/components/Transaction.svelte index 162cb9592a2..fbe65fd94c3 100644 --- a/packages/bridge-ui/src/components/Transaction.svelte +++ b/packages/bridge-ui/src/components/Transaction.svelte @@ -7,11 +7,9 @@ import { Contract, ethers } from "ethers"; import { bridges } from "../store/bridge"; import { signer } from "../store/signer"; - import { pendingTransactions, transactions } from "../store/transactions"; + import { pendingTransactions } from "../store/transactions"; import { errorToast, successToast } from "../utils/toast"; import { _ } from "svelte-i18n"; - import { switchEthereumChain } from "../utils/switchEthereumChain"; - import { ethereum } from "../store/ethereum"; import { fromChain as fromChainStore, toChain as toChainStore, @@ -22,6 +20,7 @@ import { LottiePlayer } from "@lottiefiles/svelte-lottie-player"; import HeaderSync from "../constants/abi/HeaderSync"; import { providers } from "../store/providers"; + import { fetchSigner, switchNetwork } from "@wagmi/core"; export let transaction: BridgeTransaction; @@ -36,7 +35,9 @@ async function claim(bridgeTx: BridgeTransaction) { if (fromChain.id !== bridgeTx.message.destChainId.toNumber()) { const chain = chains[bridgeTx.message.destChainId.toNumber()]; - await switchEthereumChain($ethereum, chain); + await switchNetwork({ + chainId: chain.id, + }); const provider = new ethers.providers.Web3Provider(window.ethereum); await provider.send("eth_requestAccounts", []); @@ -46,7 +47,8 @@ } else { toChainStore.set(CHAIN_MAINNET); } - signer.set(provider.getSigner()); + const wagmiSigner = await fetchSigner(); + signer.set(wagmiSigner); } try { @@ -77,7 +79,7 @@ async function isProcessable() { if (!transaction.receipt) return false; if (!transaction.message) return false; - if (transaction.status === MessageStatus.Done) return true; + if (transaction.status !== MessageStatus.New) return true; const contract = new Contract( chains[transaction.message.destChainId.toNumber()].headerSyncAddress, @@ -89,8 +91,7 @@ const srcBlock = await $providers .get(chains[transaction.message.srcChainId.toNumber()].id) .getBlock(latestSyncedHeader); - - return transaction.receipt.blockNumber <= srcBlock.number; + return transaction.receipt.blockNumber >= srcBlock.number; } diff --git a/packages/bridge-ui/src/components/buttons/Connect.svelte b/packages/bridge-ui/src/components/buttons/Connect.svelte index 30d7abf9015..c87de2b2a7e 100644 --- a/packages/bridge-ui/src/components/buttons/Connect.svelte +++ b/packages/bridge-ui/src/components/buttons/Connect.svelte @@ -1,62 +1,105 @@ - await connect()} + ($isConnectWalletModalOpen = true)} >{$_("nav.connect")} + + ($isConnectWalletModalOpen = false)} +> + + {#each $wagmiClient.connectors as connector} + connectWithConnector(connector)} + > + + + + {connector.name} + + {/each} + + diff --git a/packages/bridge-ui/src/components/form/SelectChain.svelte b/packages/bridge-ui/src/components/form/SelectChain.svelte index 8e9588befcd..b4242b5797b 100644 --- a/packages/bridge-ui/src/components/form/SelectChain.svelte +++ b/packages/bridge-ui/src/components/form/SelectChain.svelte @@ -2,16 +2,17 @@ import ArrowRightLeft from "../icons/ArrowRightLeft.svelte"; import { fromChain, toChain } from "../../store/chain"; import { CHAIN_MAINNET, CHAIN_TKO } from "../../domain/chain"; - import { ethereum } from "../../store/ethereum"; import { signer } from "../../store/signer"; - import { switchEthereumChain } from "../../utils/switchEthereumChain"; import { ethers } from "ethers"; import { errorToast, successToast } from "../../utils/toast"; + import { switchNetwork } from "@wagmi/core"; const toggleChains = async () => { try { const chain = $fromChain === CHAIN_MAINNET ? CHAIN_TKO : CHAIN_MAINNET; - await switchEthereumChain($ethereum, chain); + await switchNetwork({ + chainId: chain.id, + }); const provider = new ethers.providers.Web3Provider(window.ethereum); await provider.send("eth_requestAccounts", []); diff --git a/packages/bridge-ui/src/components/icons/CoinbaseWallet.svelte b/packages/bridge-ui/src/components/icons/CoinbaseWallet.svelte new file mode 100644 index 00000000000..297732d01c2 --- /dev/null +++ b/packages/bridge-ui/src/components/icons/CoinbaseWallet.svelte @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/packages/bridge-ui/src/components/icons/WalletConnect.svelte b/packages/bridge-ui/src/components/icons/WalletConnect.svelte new file mode 100644 index 00000000000..474cecf5e9c --- /dev/null +++ b/packages/bridge-ui/src/components/icons/WalletConnect.svelte @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/packages/bridge-ui/src/components/modals/SwitchEthereumChainModal.svelte b/packages/bridge-ui/src/components/modals/SwitchEthereumChainModal.svelte index 9251e4c2734..71da1440788 100644 --- a/packages/bridge-ui/src/components/modals/SwitchEthereumChainModal.svelte +++ b/packages/bridge-ui/src/components/modals/SwitchEthereumChainModal.svelte @@ -1,22 +1,21 @@