Finish Exchange.js

This commit is contained in:
Daniel
2022-11-05 15:24:19 +01:00
parent 3ecfaa3908
commit da3c8798be
2 changed files with 154 additions and 92 deletions

View File

@@ -7,103 +7,161 @@ import { parseUnits } from "ethers/lib/utils";
import { getAvailableTokens, getCounterpartTokens, findPoolByTokens, isOperationPending, getFailureMessage, getSuccessMessage } from '../utils'; import { getAvailableTokens, getCounterpartTokens, findPoolByTokens, isOperationPending, getFailureMessage, getSuccessMessage } from '../utils';
import { ROUTER_ADDRESS } from "../config"; import { ROUTER_ADDRESS } from "../config";
import AmountIn from "./"; import AmountIn from "./AmountIn";
import AmountOut from "./"; import AmountOut from "./AmountOut";
import Balance from "./"; import Balance from "./Balance";
import styles from "../styles"; import styles from "../styles";
const Exchange = ({ pools }) => { const Exchange = ({ pools }) => {
const { account } = useEthers(); const { account } = useEthers();
const [fromValue, setFromValue] = useState("0"); const [fromValue, setFromValue] = useState("0");
const [fromToken, setFromToken] = useState(pools[0].token0Address); const [fromToken, setFromToken] = useState(pools[0].token0Address); // initialFromToken
const [toToken, setToToken] = useState(""); const [toToken, setToToken] = useState("");
const [resetState, setResetState] = useState(false) const [resetState, setResetState] = useState(false)
const fromValueBigNumber = parseUnits(fromValue || "0"); const fromValueBigNumber = parseUnits(fromValue || "0"); // converse the string to bigNumber
const availableTokens = getAvailableTokens(pools); //What tokens can we swap from? const availableTokens = getAvailableTokens(pools);
const counterpartTokens = getCounterpartTokens(pools, fromToken); //What tokens can we swap to? const counterpartTokens = getCounterpartTokens(pools, fromToken);
const pairAddress = findPoolByTokens(pools, fromToken, toToken)?.address ?? ""; //find a pair address ot that liquidity pair const pairAddress = findPoolByTokens(pools, fromToken, toToken)?.address ?? "";
//Get contract addresses const routerContract = new Contract(ROUTER_ADDRESS, abis.router02);
const routerContract = new Contract(ROUTER_ADDRESS, abis.router02); const fromTokenContract = new Contract(fromToken, ERC20.abi);
const fromTokenContract = new Contract(fromToken, ERC20.abi); const fromTokenBalance = useTokenBalance(fromToken, account);
const fromTokenBalance = useTokenBalance(fromToken, account); //Know balance of "from Token" const toTokenBalance = useTokenBalance(toToken, account);
const toTokenBalance = useTokenBalance(toToken, account); //Know balance of "to Token" const tokenAllowance = useTokenAllowance(fromToken, account, ROUTER_ADDRESS) || parseUnits("0");
const tokenAllowance = useTokenAllowance(fromToken, account, ROUTER_ADDRESS) || parseUnits("0"); const approvedNeeded = fromValueBigNumber.gt(tokenAllowance);
const approvedNeeded = fromValueBigNumber.gt(tokenAllowance); //We need to approve to make the swap const formValueIsGreaterThan0 = fromValueBigNumber.gt(parseUnits("0"));
const formValueIsGreaterThan0 = fromValueBigNumber.gt(parseUnits("0")); //has to be greater than 0 const hasEnoughBalance = fromValueBigNumber.lte(fromTokenBalance ?? parseUnits("0"));
const hasEnoughBalance = fromValueBigNumber.lte(fromTokenBalance ?? parseUnits("0")); //lte = lower than or equal to
const isApproving = isOperationPending(swapApproveState); // approve initiating a contract call (similar to use state) -> gives the state and the sender...
const isSwapping = isOperationPending(swapExecuteState); const { state: swapApproveState, send: swapApproveSend } =
useContractFunction(fromTokenContract, "approve", {
transactionName: "onApproveRequested",
gasLimitBufferPercentage: 10,
});
// swap initiating a contract call (similar to use state) -> gives the state and the sender...
const { state: swapExecuteState, send: swapExecuteSend } =
useContractFunction(routerContract, "swapExactTokensForTokens", {
transactionName: "swapExactTokensForTokens",
gasLimitBufferPercentage: 10,
});
const successMessage = getSuccessMessage(swapApproveState, swapExecuteState); const isApproving = isOperationPending(swapApproveState);
const failureMessage = getFailureMessage(swapApproveState, swapExecuteState); const isSwapping = isOperationPending(swapExecuteState);
const canApprove = !isApproving && approvedNeeded;
const canSwap = !approvedNeeded && !isSwapping && formValueIsGreaterThan0 && hasEnoughBalance;
return ( const successMessage = getSuccessMessage(swapApproveState, swapExecuteState);
<div className="flex flex-col w-full items-center"> const failureMessage = getFailureMessage(swapApproveState, swapExecuteState);
<div className="mb-8">
<AmountIn
value={fromValue}
onChange={onFromValueChange}
currencyValue={fromToken}
onSelect={onFromTokenChange}
currencies={availableTokens}
isSwapping={isSwapping && hasEnoughBalance}
/>
<Balance tokenBalance={fromTokenBalance} />
</div>
<div className="mb-8 w-[100%]"> const onApproveRequested = () => {
<AmountOut swapApproveSend(ROUTER_ADDRESS, ethers.constants.MaxUint256);
fromToken={fromToken} };
toToken={toToken}
amountIn={fromValueBigNumber}
pairContract={pairAddress}
currencyValue={toToken}
onSelect={onToTokenChange}
currencies={counterpartTokens}
/>
<Balance tokenBalance={toTokenBalance} />
</div>
{approvedNeeded && !isSwapping ? ( // https://docs.uniswap.org/protocol/V2/reference/smart-contracts/router-02#swapexacttokensfortokens
<button const onSwapRequested = () => {
disabled={!canApprove} swapExecuteSend(
onClick={onApproveRequested} fromValueBigNumber,
className={`${ 0,
canApprove [fromToken, toToken],
? "bg-site-pink text-white" account,
: "bg-site-dim2 text-site-dim2" Math.floor(Date.now() / 1000) + 60 * 20
} ${styles.actionButton}`} ).then((_) => {
> setFromValue("0");
{isApproving ? "Approving..." : "Approve"} });
</button> };
) : (
<button
disabled={!canSwap}
onClick={onSwapRequested}
className={`${
canSwap ? "bg-site-pink text-white" : "bg-site-dim2 text-site-dim2"
} ${styles.actionButton}`}
>
{isSwapping
? "Swapping..."
: hasEnoughBalance
? "Swap"
: "Insufficient balance"}
</button>
)}
{failureMessage && !resetState ? ( const onFromValueChange = (value) => {
<p className={styles.message}>{failureMessage}</p> const trimmedValue = value.trim();
) : successMessage ? (
<p className={styles.message}>{successMessage}</p> try {
) : ( trimmedValue && parseUnits(value);
"" setFromValue(value);
)} } catch (e) {}
</div> };
);
} const onFromTokenChange = (value) => {
setFromToken(value);
};
const onToTokenChange = (value) => {
setToToken(value);
};
useEffect(() => {
if(failureMessage || successMessage) {
setTimeout(() => {
setResetState(true)
setFromValue("0")
setToToken("")
}, 5000)
}
}, [failureMessage, successMessage])
return (
<div className="flex flex-col w-full items-center">
<div className="mb-8">
<AmountIn
value={fromValue}
onChange={onFromValueChange}
currencyValue={fromToken}
onSelect={onFromTokenChange}
currencies={availableTokens}
isSwapping={isSwapping && hasEnoughBalance}
/>
<Balance tokenBalance={fromTokenBalance} />
</div>
<div className="mb-8 w-[100%]">
<AmountOut
fromToken={fromToken}
toToken={toToken}
amountIn={fromValueBigNumber}
pairContract={pairAddress}
currencyValue={toToken}
onSelect={onToTokenChange}
currencies={counterpartTokens}
/>
<Balance tokenBalance={toTokenBalance} />
</div>
{approvedNeeded && !isSwapping ? (
<button
disabled={!canApprove}
onClick={onApproveRequested}
className={`${
canApprove
? "bg-site-pink text-white"
: "bg-site-dim2 text-site-dim2"
} ${styles.actionButton}`}
>
{isApproving ? "Approving..." : "Approve"}
</button>
) : (
<button
disabled={!canSwap}
onClick={onSwapRequested}
className={`${
canSwap ? "bg-site-pink text-white" : "bg-site-dim2 text-site-dim2"
} ${styles.actionButton}`}
>
{isSwapping
? "Swapping..."
: hasEnoughBalance
? "Swap"
: "Insufficient balance"}
</button>
)}
{failureMessage && !resetState ? (
<p className={styles.message}>{failureMessage}</p>
) : successMessage ? (
<p className={styles.message}>{successMessage}</p>
) : (
""
)}
</div>
);
};
export default Exchange; export default Exchange;

View File

@@ -6,7 +6,11 @@ import {ethereumLogo2 } from "../assets";
const Loader = ({ title }) => { const Loader = ({ title }) => {
return ( return (
<div className={styles.loader}> <div className={styles.loader}>
<img src={ethereumLogo2} alt="Ethereum Logo" className={styles.loaderImg} /> <img
src={ethereumLogo2}
alt="Ethereum Logo"
className={styles.loaderImg}
/>
<p className={styles.loaderText}>{title}</p> <p className={styles.loaderText}>{title}</p>
</div> </div>