diff --git a/src/Content.tsx b/src/Content.tsx index 2fcadd3..c07591c 100644 --- a/src/Content.tsx +++ b/src/Content.tsx @@ -1,7 +1,9 @@ import { Route, Routes } from 'react-router-dom' import { Home } from '@/pages/Home' - +import { toast } from 'sonner' +import { useEffect } from 'react' +import { useNetwork } from './contexts/NetworkContext' const pages = [ { path: '', @@ -14,6 +16,10 @@ const pages = [ ] export const Content = () => { + const { lightClientLoaded, isLight } = useNetwork() + useEffect(() => { + isLight && lightClientLoaded && toast.success('Light client: Synced') + }, [isLight, lightClientLoaded]) return ( <> diff --git a/src/contexts/NetworkContext.tsx b/src/contexts/NetworkContext.tsx index 6be0e52..941aa77 100644 --- a/src/contexts/NetworkContext.tsx +++ b/src/contexts/NetworkContext.tsx @@ -4,6 +4,10 @@ import { dot, ksm } from '@polkadot-api/descriptors' import { PolkadotClient, TypedApi, createClient } from 'polkadot-api' import { getWsProvider } from 'polkadot-api/ws-provider/web' +import { getSmProvider } from 'polkadot-api/sm-provider' +import SmWorker from 'polkadot-api/smoldot/worker?worker' +import { startFromWorker } from 'polkadot-api/smoldot/from-worker' + type NetworkContextProps = { children: React.ReactNode | React.ReactNode[] } @@ -11,10 +15,12 @@ type NetworkContextProps = { // const polakdotEndpoints = ['wss://rpc.ibp.network/polkadot'] // const kusamaEndpoints = ['wss://rpc.ibp.network/kusama'] -export type NetworkProps = 'polkadot' | 'kusama' +export type NetworkProps = 'polkadot' | 'kusama' | 'polkadot-lc' | 'kusama-lc' export type ApiType = TypedApi export interface INetworkContext { + lightClientLoaded: boolean + isLight: boolean network: NetworkProps setNetwork: React.Dispatch> client: PolkadotClient | undefined @@ -24,6 +30,8 @@ export interface INetworkContext { const NetworkContext = createContext(undefined) const NetworkContextProvider = ({ children }: NetworkContextProps) => { + const [lightClientLoaded, setLightClientLoaded] = useState(false) + const [isLight, setIsLight] = useState(false) const [client, setClient] = useState() const [api, setApi] = useState() const [network, setNetwork] = useState('polkadot') @@ -31,19 +39,54 @@ const NetworkContextProvider = ({ children }: NetworkContextProps) => { useEffect(() => { let cl: PolkadotClient let typedApi: ApiType - if (network === 'polkadot') { - cl = createClient(getWsProvider('wss://rpc.ibp.network/polkadot')) - typedApi = cl.getTypedApi(dot) - } else { - cl = createClient(getWsProvider('wss://rpc.ibp.network/kusama')) - typedApi = cl.getTypedApi(ksm) + switch (network) { + case 'kusama': + setIsLight(false) + cl = createClient(getWsProvider('wss://rpc.ibp.network/kusama')) + typedApi = cl.getTypedApi(ksm) + break + case 'polkadot-lc': { + setIsLight(true) + const smoldot = startFromWorker(new SmWorker()) + const dotRelayChain = import('polkadot-api/chains/polkadot').then( + ({ chainSpec }) => smoldot.addChain({ chainSpec }), + ) + cl = createClient(getSmProvider(dotRelayChain)) + typedApi = cl.getTypedApi(dot) + break + } + case 'kusama-lc': { + setIsLight(true) + const smoldot = startFromWorker(new SmWorker()) + const ksmRelayChain = import('polkadot-api/chains/ksmcc3').then( + ({ chainSpec }) => smoldot.addChain({ chainSpec }), + ) + cl = createClient(getSmProvider(ksmRelayChain)) + typedApi = cl.getTypedApi(ksm) + break + } + default: + setIsLight(false) + cl = createClient(getWsProvider('wss://rpc.ibp.network/polkadot')) + typedApi = cl.getTypedApi(dot) } setClient(cl) setApi(typedApi) }, [network]) + useEffect(() => { + isLight && + client?.finalizedBlock$.subscribe((finalizedBlock) => { + if (finalizedBlock.number && !lightClientLoaded) { + setLightClientLoaded(true) + } + }) + }, [client?.finalizedBlock$, isLight, lightClientLoaded]) + return ( - + {children} ) diff --git a/src/header.tsx b/src/header.tsx index 4bd5437..b2d8ad2 100644 --- a/src/header.tsx +++ b/src/header.tsx @@ -10,7 +10,7 @@ import { Sheet, SheetContent, SheetTrigger } from '@/components/ui/sheet' import { Button } from '@/components/ui/button' import { routes } from '@/lib/utils' import { useWalletDisconnector } from '@reactive-dot/react' -import { Settings2, PanelLeft } from 'lucide-react' +import { PanelLeft } from 'lucide-react' // import { // Menubar, @@ -23,7 +23,14 @@ import { Settings2, PanelLeft } from 'lucide-react' // } from '@/components/ui/menubar' import { useAccounts } from './contexts/AccountsContext' import { useEffect } from 'react' -import { useNetwork } from './contexts/NetworkContext' +import { NetworkProps, useNetwork } from './contexts/NetworkContext' + +const networkList = [ + 'Polkadot|polkadot', + 'Polkadot Light Client|polkadot-lc', + 'Kusama|kusama', + 'Kusama Light Client|kusama-lc', +] export const Header = () => { const { network, setNetwork } = useNetwork() @@ -57,13 +64,6 @@ export const Header = () => { {r.name} ))} - - - Settings - @@ -94,20 +94,15 @@ export const Header = () => { - setNetwork('polkadot')} - > - Polkadot - - setNetwork('kusama')} - > - Kusama - + {networkList.map((n) => ( + setNetwork(n.split('|')[1] as NetworkProps)} + > + {n.split('|')[0]} + + ))} {!accounts.length && ( diff --git a/src/navigation.tsx b/src/navigation.tsx index 53a5cc4..2f3e687 100644 --- a/src/navigation.tsx +++ b/src/navigation.tsx @@ -7,10 +7,13 @@ import { import { routes } from '@/lib/utils' import { useLocation } from 'react-router-dom' import PolkadotIcon from '@/assets/img/polkadotIcon.svg?react' +import { TbLoaderQuarter } from 'react-icons/tb' +import { FaCheckCircle } from 'react-icons/fa' import { Github, Moon, Sun } from 'lucide-react' import { Button } from '@/components/ui/button' import { useTheme } from '@/components/theme-provider' +import { useNetwork } from './contexts/NetworkContext' const linkStyle = (pathname: string, link: string) => { return `link ${ @@ -21,6 +24,7 @@ const linkStyle = (pathname: string, link: string) => { } export const Navigation = () => { + const { lightClientLoaded, isLight } = useNetwork() const { pathname } = useLocation() const { theme, setTheme } = useTheme() @@ -51,6 +55,28 @@ export const Navigation = () => { ))} )