import { Decimal } from "@cosmjs/math"; import { MsgSendEncodeObject } from "@cosmjs/stargate"; import { assert } from "@cosmjs/utils"; import { useEffect, useState } from "react"; import { MsgGetter } from ".."; import { useAppContext } from "../../../../context/AppContext"; import { checkAddress, exampleAddress } from "../../../../lib/displayHelpers"; import { ChainInfo } from "../../../../types"; import { MsgCodecs, MsgTypeUrls } from "../../../../types/txMsg"; import Input from "../../../inputs/Input"; import Select from "../../../inputs/Select"; import StackableContainer from "../../../layout/StackableContainer"; const getDenomOptions = (assets: ChainInfo["assets"]) => { const customDenomOption = { label: "Custom (enter denom below)", value: "custom" }; if (!assets?.length) { return [customDenomOption]; } const filteredAssets = assets.filter(({ denom: denomToFilter }) => { if (denomToFilter.startsWith("u")) { const foundMacroDenom = assets.find( ({ denom }) => denom.toLowerCase() === denomToFilter.slice(1).toLowerCase(), ); if (foundMacroDenom) { return false; } } return true; }); const denomOptions = filteredAssets.map(({ denom }) => ({ label: denom.toUpperCase(), value: denom, })); return [...denomOptions, customDenomOption]; }; interface MsgSendFormProps { readonly fromAddress: string; readonly setMsgGetter: (msgGetter: MsgGetter) => void; readonly deleteMsg: () => void; } const MsgSendForm = ({ fromAddress, setMsgGetter, deleteMsg }: MsgSendFormProps) => { const { state } = useAppContext(); assert(state.chain.addressPrefix, "addressPrefix missing"); const denomOptions = getDenomOptions(state.chain.assets); const [toAddress, setToAddress] = useState(""); const [selectedDenom, setSelectedDenom] = useState(denomOptions[0]); const [customDenom, setCustomDenom] = useState(""); const [amount, setAmount] = useState("0"); const [toAddressError, setToAddressError] = useState(""); const [customDenomError, setCustomDenomError] = useState(""); const [amountError, setAmountError] = useState(""); useEffect(() => { assert(state.chain.denom, "denom missing"); setToAddressError(""); setCustomDenomError(""); setAmountError(""); const isMsgValid = (): boolean => { assert(state.chain.addressPrefix, "addressPrefix missing"); const addressErrorMsg = checkAddress(toAddress, state.chain.addressPrefix); if (addressErrorMsg) { setToAddressError(`Invalid address for network ${state.chain.chainId}: ${addressErrorMsg}`); return false; } if (selectedDenom.value === "custom" && !customDenom) { setCustomDenomError("Custom denom must be set because of selection above"); return false; } if (!amount || Number(amount) <= 0) { setAmountError("Amount must be greater than 0"); return false; } if (selectedDenom.value === "custom" && !Number.isInteger(amount)) { setAmountError("Amount cannot be decimal for custom denom"); return false; } return true; }; const denom = ( selectedDenom.value === "custom" ? customDenom : selectedDenom.value ).toLowerCase(); const exponent = state.chain.assets?.find(({ denom: currentDenom }) => currentDenom.toLowerCase() === denom) ?.exponent ?? 0; const amountInAtomics = (() => { try { return Decimal.fromUserInput(amount, exponent).atomics; } catch { return "0"; } })(); const msgValue = MsgCodecs[MsgTypeUrls.Send].fromPartial({ fromAddress, toAddress, amount: [{ denom, amount: amountInAtomics }], }); const msg: MsgSendEncodeObject = { typeUrl: MsgTypeUrls.Send, value: msgValue }; setMsgGetter({ isMsgValid, msg }); }, [ amount, customDenom, fromAddress, selectedDenom.value, setMsgGetter, state.chain.addressPrefix, state.chain.assets, state.chain.chainId, state.chain.denom, toAddress, ]); return (

MsgSend

setToAddress(target.value)} error={toAddressError} placeholder={`E.g. ${exampleAddress(0, state.chain.addressPrefix)}`} />
setCustomDenom(target.value)} placeholder={ selectedDenom.value === "custom" ? "Enter custom denom" : "Select Custom denom above" } disabled={selectedDenom.value !== "custom"} error={customDenomError} />
setAmount(target.value)} error={amountError} />
); }; export default MsgSendForm;