174 lines
5.2 KiB
TypeScript
174 lines
5.2 KiB
TypeScript
import React, { useEffect, useState, useMemo, useContext } from "react";
|
|
import { useParams, useLocation } from "react-router";
|
|
import { ethers } from "ethers";
|
|
import queryString from "query-string";
|
|
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 } from "./types";
|
|
import { PAGE_SIZE } from "./params";
|
|
import { useFeeToggler } from "./search/useFeeToggler";
|
|
import { RuntimeContext } from "./useRuntime";
|
|
import { useENSCache } from "./useReverseCache";
|
|
|
|
type BlockParams = {
|
|
blockNumber: string;
|
|
};
|
|
|
|
type PageParams = {
|
|
p?: number;
|
|
};
|
|
|
|
const BlockTransactions: React.FC = () => {
|
|
const { provider } = useContext(RuntimeContext);
|
|
const params = useParams<BlockParams>();
|
|
const location = useLocation<PageParams>();
|
|
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<ProcessedTransaction[]>();
|
|
useEffect(() => {
|
|
if (!provider) {
|
|
return;
|
|
}
|
|
|
|
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`;
|
|
|
|
const responses = _block.transactions
|
|
.map((t, i): ProcessedTransaction => {
|
|
return {
|
|
blockNumber: blockNumber.toNumber(),
|
|
timestamp: _block.timestamp,
|
|
miner: _block.miner,
|
|
idx: i,
|
|
hash: t.hash,
|
|
from: t.from,
|
|
to: t.to,
|
|
value: t.value,
|
|
fee: provider.formatter
|
|
.bigNumber(_receipts[i].gasUsed)
|
|
.mul(t.gasPrice!),
|
|
gasPrice: t.gasPrice!,
|
|
data: t.data,
|
|
status: provider.formatter.number(_receipts[i].status),
|
|
};
|
|
})
|
|
.reverse();
|
|
setTxs(responses);
|
|
|
|
const internalChecks = await Promise.all(
|
|
responses.map(async (res) => {
|
|
const r = await provider.send("ots_getTransactionTransfers", [
|
|
res.hash,
|
|
]);
|
|
for (const t of r) {
|
|
if (
|
|
res.miner &&
|
|
(res.miner === ethers.utils.getAddress(t.from) ||
|
|
res.miner === ethers.utils.getAddress(t.to))
|
|
) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
})
|
|
);
|
|
const processedResponses = responses.map((r, i): ProcessedTransaction => {
|
|
return { ...r, internalMinerInteraction: internalChecks[i] };
|
|
});
|
|
setTxs(processedResponses);
|
|
};
|
|
readBlock();
|
|
}, [provider, 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 = useENSCache(provider, page);
|
|
|
|
document.title = `Block #${blockNumber} Txns | Otterscan`;
|
|
|
|
const [feeDisplay, feeDisplayToggler] = useFeeToggler();
|
|
|
|
return (
|
|
<StandardFrame>
|
|
<StandardSubtitle>Transactions</StandardSubtitle>
|
|
<div className="pb-2 text-sm text-gray-500">
|
|
For Block <BlockLink blockTag={blockNumber.toNumber()} />
|
|
</div>
|
|
<ContentFrame>
|
|
<div className="flex justify-between items-baseline py-3">
|
|
<div className="text-sm text-gray-500">
|
|
{page === undefined ? (
|
|
<>Waiting for search results...</>
|
|
) : (
|
|
<>A total of {total} transactions found</>
|
|
)}
|
|
</div>
|
|
<PageControl
|
|
pageNumber={pageNumber}
|
|
pageSize={PAGE_SIZE}
|
|
total={total}
|
|
/>
|
|
</div>
|
|
<ResultHeader
|
|
feeDisplay={feeDisplay}
|
|
feeDisplayToggler={feeDisplayToggler}
|
|
/>
|
|
{page ? (
|
|
<>
|
|
{page.map((tx) => (
|
|
<TransactionItem
|
|
key={tx.hash}
|
|
tx={tx}
|
|
ensCache={reverseCache}
|
|
feeDisplay={feeDisplay}
|
|
/>
|
|
))}
|
|
<div className="flex justify-between items-baseline py-3">
|
|
<div className="text-sm text-gray-500">
|
|
A total of {total} transactions found
|
|
</div>
|
|
<PageControl
|
|
pageNumber={pageNumber}
|
|
pageSize={PAGE_SIZE}
|
|
total={total}
|
|
/>
|
|
</div>
|
|
</>
|
|
) : (
|
|
<PendingResults />
|
|
)}
|
|
</ContentFrame>
|
|
</StandardFrame>
|
|
);
|
|
};
|
|
|
|
export default React.memo(BlockTransactions);
|