import React, { useEffect, useState, useMemo } from "react"; import { useParams, useLocation } from "react-router"; import { ethers } from "ethers"; import queryString from "query-string"; import { provider } from "./ethersconfig"; import StandardFrame from "./StandardFrame"; import StandardSubtitle from "./StandardSubtitle"; import ContentFrame from "./ContentFrame"; import PageControl from "./search/PageControl"; import ResultHeader from "./search/ResultHeader"; import PendingResults from "./search/PendingResults"; import TransactionItem from "./search/TransactionItem"; import BlockLink from "./components/BlockLink"; import { ProcessedTransaction, ENSReverseCache } from "./types"; import { PAGE_SIZE } from "./params"; import { useFeeToggler } from "./search/useFeeToggler"; type BlockParams = { blockNumber: string; }; type PageParams = { p?: number; }; const BlockTransactions: React.FC = () => { const params = useParams(); const location = useLocation(); const qs = queryString.parse(location.search); let pageNumber = 1; if (qs.p) { try { pageNumber = parseInt(qs.p as string); } catch (err) {} } const blockNumber = useMemo( () => ethers.BigNumber.from(params.blockNumber), [params.blockNumber] ); const [txs, setTxs] = useState(); useEffect(() => { const readBlock = async () => { const [_block, _receipts] = await Promise.all([ provider.getBlockWithTransactions(blockNumber.toNumber()), provider.send("eth_getBlockReceipts", [blockNumber.toNumber()]), ]); document.title = `Block #${_block.number} Transactions | Otterscan`; setTxs( _block.transactions .map((t, i) => { return { blockNumber: blockNumber.toNumber(), timestamp: _block.timestamp, idx: i, hash: t.hash, from: t.from, to: t.to, value: t.value, fee: t.gasLimit.mul(t.gasPrice!), gasPrice: t.gasPrice!, data: t.data, status: _receipts[i].status, }; }) .reverse() ); }; readBlock(); }, [blockNumber]); const page = useMemo(() => { if (!txs) { return undefined; } const pageStart = (pageNumber - 1) * PAGE_SIZE; return txs.slice(pageStart, pageStart + PAGE_SIZE); }, [txs, pageNumber]); const total = useMemo(() => txs?.length ?? 0, [txs]); const [reverseCache, setReverseCache] = useState(); useEffect(() => { if (!page) { return; } const addrSet = new Set(); for (const tx of page) { if (tx.from) { addrSet.add(tx.from); } if (tx.to) { addrSet.add(tx.to); } } const addresses = Array.from(addrSet); const reverseResolve = async () => { const solvers: Promise[] = []; for (const a of addresses) { solvers.push(provider.lookupAddress(a)); } const results = await Promise.all(solvers); const cache: ENSReverseCache = {}; for (let i = 0; i < results.length; i++) { if (results[i] === null) { continue; } cache[addresses[i]] = results[i]; } console.log("RESOLVED"); console.log(cache); setReverseCache(cache); }; reverseResolve(); }, [page]); document.title = `Block #${blockNumber} Txns | Otterscan`; const [feeDisplay, feeDisplayToggler] = useFeeToggler(); return ( Transactions
For Block
{page === undefined ? ( <>Waiting for search results... ) : ( <>A total of {total} transactions found )}
{page ? ( <> {page.map((tx) => ( ))}
A total of {total} transactions found
) : ( )}
); }; export default React.memo(BlockTransactions);