otterscan/src/transaction/TraceInput.tsx

114 lines
3.5 KiB
TypeScript
Raw Normal View History

2021-11-06 18:54:34 +00:00
import React, { useState } from "react";
import { Switch } from "@headlessui/react";
2021-11-21 11:52:37 +00:00
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
import { faBomb } from "@fortawesome/free-solid-svg-icons/faBomb";
2021-11-07 20:46:42 +00:00
import TransactionAddress from "../components/TransactionAddress";
2021-11-06 05:12:37 +00:00
import FormattedBalance from "../components/FormattedBalance";
import FunctionSignature from "./FunctionSignature";
2021-11-08 11:28:28 +00:00
import InputDecoder from "./decoder/InputDecoder";
2021-11-06 05:12:37 +00:00
import { TraceEntry } from "../useErigonHooks";
import { ResolvedAddresses } from "../api/address-resolver";
2021-11-06 18:54:34 +00:00
import {
extract4Bytes,
FourBytesEntry,
useTransactionDescription,
} from "../use4Bytes";
2021-11-06 05:12:37 +00:00
type TraceInputProps = {
t: TraceEntry;
fourBytesMap: Record<string, FourBytesEntry | null | undefined>;
resolvedAddresses: ResolvedAddresses | undefined;
};
const TraceInput: React.FC<TraceInputProps> = ({
t,
fourBytesMap,
resolvedAddresses,
}) => {
const raw4Bytes = extract4Bytes(t.input);
2021-11-06 18:54:34 +00:00
const fourBytes = raw4Bytes !== null ? fourBytesMap[raw4Bytes] : null;
2021-11-06 05:12:37 +00:00
const sigText =
2021-11-06 18:54:34 +00:00
raw4Bytes === null ? "<fallback>" : fourBytes?.name ?? raw4Bytes;
const hasParams = t.input.length > 10;
const fourBytesTxDesc = useTransactionDescription(
fourBytes,
t.input,
t.value
);
const [expanded, setExpanded] = useState<boolean>(false);
2021-11-06 05:12:37 +00:00
return (
2021-11-08 18:20:13 +00:00
<div
className={`ml-5 border hover:border-gray-500 rounded px-1 py-0.5 ${
expanded ? "w-full" : ""
}`}
>
2021-11-06 18:54:34 +00:00
<div className="flex items-baseline">
<span className="text-xs text-gray-400 lowercase">{t.type}</span>
2021-11-22 17:16:33 +00:00
{t.type === "SELFDESTRUCT" ? (
<span className="pl-2 text-red-800" title="Self destruct">
<FontAwesomeIcon icon={faBomb} size="1x" />
</span>
) : (
2021-11-21 11:52:37 +00:00
<>
2021-11-22 17:16:33 +00:00
<span>
<TransactionAddress
address={t.to}
resolvedAddresses={resolvedAddresses}
/>
2021-11-21 11:52:37 +00:00
</span>
2021-11-22 17:16:33 +00:00
{t.type !== "CREATE" && t.type !== "CREATE2" && (
<>
<span>.</span>
<FunctionSignature callType={t.type} sig={sigText} />
{t.value && !t.value.isZero() && (
<span className="text-red-700 whitespace-nowrap">
{"{"}value: <FormattedBalance value={t.value} /> ETH{"}"}
</span>
)}
<span className="whitespace-nowrap">
(
{hasParams && (
<Switch
className="text-xs"
checked={expanded}
onChange={setExpanded}
>
{expanded ? (
<span className="text-gray-400">[-]</span>
) : (
<>[...]</>
)}
</Switch>
)}
2021-11-22 17:16:33 +00:00
{(!hasParams || !expanded) && <>)</>}
</span>
</>
)}
</>
)}
2021-11-06 18:54:34 +00:00
</div>
{hasParams && expanded && (
2021-11-06 18:54:34 +00:00
<>
2021-11-08 00:07:20 +00:00
<div className="ml-5 mr-1 my-2">
2021-11-08 11:28:28 +00:00
<InputDecoder
fourBytes={raw4Bytes ?? "0x"}
resolvedTxDesc={fourBytesTxDesc}
2021-11-06 18:54:34 +00:00
hasParamNames={false}
2021-11-08 11:28:28 +00:00
data={t.input}
userMethod={undefined}
devMethod={undefined}
2021-11-06 18:54:34 +00:00
resolvedAddresses={resolvedAddresses}
/>
</div>
<div>)</div>
</>
2021-11-06 05:12:37 +00:00
)}
</div>
);
};
export default TraceInput;