import { MsgTransferEncodeObject } from "@cosmjs/stargate"; import { useEffect, useState } from "react"; import { MsgGetter } from ".."; import { useChains } from "../../../../context/ChainsContext"; import { datetimeLocalFromTimestamp, timestampFromDatetimeLocal, } from "../../../../lib/dateHelpers"; import { checkAddress, exampleAddress, trimStringsObj } from "../../../../lib/displayHelpers"; import { MsgCodecs, MsgTypeUrls } from "../../../../types/txMsg"; import Input from "../../../inputs/Input"; import StackableContainer from "../../../layout/StackableContainer"; const humanTimestampOptions = [ { label: "12 hours from now", value: 12 * 60 * 60 * 1000 }, { label: "1 day from now", value: 24 * 60 * 60 * 1000 }, { label: "2 days from now", value: 2 * 24 * 60 * 60 * 1000 }, { label: "3 days from now", value: 3 * 24 * 60 * 60 * 1000 }, { label: "7 days from now", value: 7 * 24 * 60 * 60 * 1000 }, { label: "10 days from now", value: 10 * 24 * 60 * 60 * 1000 }, { label: "2 weeks from now", value: 2 * 7 * 24 * 60 * 60 * 1000 }, { label: "3 weeks from now", value: 3 * 7 * 24 * 60 * 60 * 1000 }, { label: "1 month from now", value: 30 * 24 * 60 * 60 * 1000 }, ]; interface MsgTransferFormProps { readonly senderAddress: string; readonly setMsgGetter: (msgGetter: MsgGetter) => void; readonly deleteMsg: () => void; } const MsgTransferForm = ({ senderAddress, setMsgGetter, deleteMsg }: MsgTransferFormProps) => { const { chain } = useChains(); const [toAddress, setToAddress] = useState(""); const [denom, setDenom] = useState(""); const [amount, setAmount] = useState("0"); const [sourcePort, setSourcePort] = useState("transfer"); const [sourceChannel, setSourceChannel] = useState(""); const [timeout, setTimeout] = useState( datetimeLocalFromTimestamp(Date.now() + humanTimestampOptions[0].value), ); const [memo, setMemo] = useState(""); const [toAddressError, setToAddressError] = useState(""); const [denomError, setDenomError] = useState(""); const [amountError, setAmountError] = useState(""); const [sourcePortError, setSourcePortError] = useState(""); const [sourceChannelError, setSourceChannelError] = useState(""); const [timeoutError, setTimeoutError] = useState(""); const trimmedInputs = trimStringsObj({ toAddress, denom, amount, sourcePort, sourceChannel, timeout, memo, }); useEffect(() => { // eslint-disable-next-line no-shadow const { toAddress, denom, amount, sourcePort, sourceChannel, timeout, memo } = trimmedInputs; const isMsgValid = (): boolean => { setToAddressError(""); setDenomError(""); setAmountError(""); setSourcePortError(""); setSourceChannelError(""); setTimeoutError(""); const addressErrorMsg = checkAddress(toAddress, null); // Allow address from any chain if (addressErrorMsg) { setToAddressError(`Invalid address for network ${chain.chainId}: ${addressErrorMsg}`); return false; } if (!denom) { setDenomError("Denom is required"); return false; } if (!amount || Number(amount) <= 0) { setAmountError("Amount must be greater than 0"); return false; } if (!sourcePort) { setSourcePortError("Source port is required"); return false; } if (!sourceChannel) { setSourceChannelError("Source channel is required"); return false; } const timeoutDate = new Date(Number(timestampFromDatetimeLocal(timeout, "ms"))); if (timeoutDate <= new Date()) { setTimeoutError("Timeout must be a date in the future"); return false; } return true; }; const msgValue = MsgCodecs[MsgTypeUrls.Transfer].fromPartial({ sender: senderAddress, receiver: toAddress, token: { denom, amount }, sourcePort, sourceChannel, timeoutTimestamp: timestampFromDatetimeLocal(timeout, "ns"), memo, }); const msg: MsgTransferEncodeObject = { typeUrl: MsgTypeUrls.Transfer, value: msgValue }; setMsgGetter({ isMsgValid, msg }); }, [chain.chainId, senderAddress, setMsgGetter, trimmedInputs]); useEffect(() => { if (!denom || !denom.startsWith("ibc/")) { return; } const foundDenom = chain.assets.find((asset) => asset.base === denom); if (!foundDenom) { return; } const trace = foundDenom.traces?.[0]; if (!trace) { return; } setSourcePort(trace.chain?.path?.split("/")?.[0] || "transfer"); setSourceChannel(trace.chain?.channel_id || ""); }, [chain.assets, denom]); return (

MsgTransfer

{ setToAddress(target.value); setToAddressError(""); }} error={toAddressError} placeholder={`E.g. ${exampleAddress(0, chain.addressPrefix)}`} />
{ setDenom(target.value); setDenomError(""); }} error={denomError} />
{ setAmount(target.value); setAmountError(""); }} error={amountError} />
{ setSourcePort(target.value); setSourcePortError(""); }} error={sourcePortError} />
{ setSourceChannel(target.value); setSourceChannelError(""); }} error={sourceChannelError} />
{ setTimeout(target.value); setTimeoutError(""); }} error={timeoutError} /> {humanTimestampOptions.map(({ label, value }) => ( ))}
setMemo(target.value)} />
); }; export default MsgTransferForm;