Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: [GSW-468] Implements Remove Position #252

Merged
merged 1 commit into from
Dec 14, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -5,19 +5,18 @@ import React from "react";
import { HeaderWrapper } from "./MyLiquidityHeader.styles";

interface MyLiquidityHeaderProps {
connected: boolean;
isSwitchNetwork: boolean;
availableRemovePosition: boolean;
handleClickAddPosition: () => void;
handleClickRemovePosition: () => void;
}

const MyLiquidityHeader: React.FC<MyLiquidityHeaderProps> = ({ connected, isSwitchNetwork, handleClickAddPosition, handleClickRemovePosition }) => {
const MyLiquidityHeader: React.FC<MyLiquidityHeaderProps> = ({ availableRemovePosition, handleClickAddPosition, handleClickRemovePosition }) => {
return (
<HeaderWrapper>
<h2>My Positions</h2>
<div className="button-wrap">
<Button
disabled={!connected || isSwitchNetwork}
disabled={!availableRemovePosition}
text="Remove Position"
onClick={handleClickRemovePosition}
style={{
Expand Down
5 changes: 3 additions & 2 deletions packages/web/src/components/pool/my-liquidity/MyLiquidity.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ interface MyLiquidityProps {
onScroll: () => void;
currentIndex: number;
claimAll: () => void;
availableRemovePosition: boolean;
}

const MyLiquidity: React.FC<MyLiquidityProps> = ({
Expand All @@ -30,13 +31,13 @@ const MyLiquidity: React.FC<MyLiquidityProps> = ({
onScroll,
currentIndex,
claimAll,
availableRemovePosition,
}) => {
return (
<MyLiquidityWrapper>
<div className="liquidity-wrap">
<MyLiquidityHeader
connected={connected}
isSwitchNetwork={isSwitchNetwork}
availableRemovePosition={availableRemovePosition}
handleClickAddPosition={handleClickAddPosition}
handleClickRemovePosition={handleClickRemovePosition}
/>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,107 +1,107 @@
import DoubleLogo from "@components/common/double-logo/DoubleLogo";
import Tooltip from "@components/common/tooltip/Tooltip";
import React, { useCallback, useMemo, useRef, useState, useEffect } from "react";
import React, { useMemo } from "react";
import { RemoveLiquiditySelectListItemWrapper, TooltipWrapperContent } from "./RemoveLiquiditySelectListItem.styles";
import Badge, { BADGE_TYPE } from "@components/common/badge/Badge";
import { convertLiquidity } from "@utils/stake-position-utils";
import { PoolPositionModel } from "@models/position/pool-position-model";
import { tooltipWrapper } from "@components/stake/select-lilquidity-list-item/SelectLiquidityListItem.styles";
import { makeDisplayTokenAmount } from "@utils/token-utils";
import { numberToUSD } from "@utils/number-utils";
import { SwapFeeTierInfoMap } from "@constants/option.constant";
import { makeSwapFeeTier } from "@utils/swap-utils";

interface RemoveLiquiditySelectListItemProps {
selected: boolean;
position: PoolPositionModel;
select: (id: string) => void;
width: number;
checkedList: string[];
onCheckedItem: (checked: boolean, path: string) => void;
disabled?: boolean;
}

interface TooltipProps {
selectable: boolean;
position: PoolPositionModel;
disabled: boolean;
}

const TooltipContent: React.FC<TooltipProps> = ({ selectable }) => {
const TooltipContent: React.FC<TooltipProps> = ({ position, disabled }) => {
return (
<TooltipWrapperContent>
<div>
<div className="title">Token ID</div>
<div className="title">#982932</div>
</div>
<div>
<div className="value">
<img src="https://raw.githubusercontent.com/Uniswap/assets/master/blockchains/ethereum/assets/0x2b591e99afE9f32eAA6214f7B7629768c40Eeb39/logo.png" />
GNS
<div css={tooltipWrapper()}>
<div>
<div className="title">Token ID</div>
<div className="title">#{position.id}</div>
</div>
<div className="value">50.05881</div>
</div>
<div>
<div className="value">
<img src="https://raw.githubusercontent.com/Uniswap/assets/master/blockchains/ethereum/assets/0x2b591e99afE9f32eAA6214f7B7629768c40Eeb39/logo.png" />
GNS
<div>
<div className="value">
<img src={position.pool.tokenA.logoURI} alt="token logo" />
{position.pool.tokenA.symbol}
</div>
<div className="value">{makeDisplayTokenAmount(position.pool.tokenA, position.token0Balance)}</div>
</div>
<div>
<div className="value">
<img src={position.pool.tokenB.logoURI} alt="token logo" />
{position.pool.tokenB.symbol}
</div>
<div className="value">{makeDisplayTokenAmount(position.pool.tokenB, position.token1Balance)}</div>
</div>
<div className="value">50.05881</div>
</div>
{selectable && <div className="divider"></div>}
{selectable && <div className="unstake-description">
*You need to unstake your position first.
</div>}
{!disabled && <div className="divider"></div>}
{!disabled && (
<div className="unstake-description">
*You need to unstake your position first.
</div>
)}
</TooltipWrapperContent>
);
};

const RemoveLiquiditySelectListItem: React.FC<RemoveLiquiditySelectListItemProps> = ({
selected,
position,
select,
width,
checkedList,
onCheckedItem,
disabled = false,
}) => {
const [checkWidth, setIsCheckWidth] = useState(true);
const leftDivRef = useRef<HTMLDivElement>(null);
const liquidityRef = useRef<HTMLDivElement>(null);
const checked = useMemo(() => {
return checkedList.includes(position.id);
}, [checkedList, position.id]);

const selectable = useMemo(() => {
return position.unclaimedFee0Amount + position.unclaimedFee1Amount > 0;
}, [position]);
const tokenA = useMemo(() => {
return position.pool.tokenA;
}, [position.pool.tokenA]);

const doubleLogo = useMemo(() => {
const { tokenA, tokenB } = position.pool;
return {
left: tokenA.logoURI,
right: tokenB.logoURI,
};
}, [position]);
const tokenB = useMemo(() => {
return position.pool.tokenB;
}, [position.pool.tokenB]);

const onChangeCheckbox = useCallback(() => {
select(position.id);
}, [position.id, select]);
const liquidityUSD = useMemo(() => {
return numberToUSD(Number(position.positionUsdValue));
}, [position.positionUsdValue]);

useEffect(() => {
if (typeof window !== "undefined") {
const windowWidth = Math.min(width, 500);
const totalWidth = (leftDivRef?.current?.offsetWidth || 0) + (liquidityRef?.current?.offsetWidth || 0) + 100;
setIsCheckWidth(windowWidth > totalWidth);
}
}, [liquidityRef.current, leftDivRef.current, width]);
const feeStr = useMemo(() => {
return SwapFeeTierInfoMap[makeSwapFeeTier(position.pool.fee)].rateStr;
}, [position]);

return (
<RemoveLiquiditySelectListItemWrapper selected={selected}>
<div className="left-content" ref={leftDivRef}>
<RemoveLiquiditySelectListItemWrapper selected={checked}>
<div className="left-content" >
<input
id={`checkbox-item-${position.id}`}
type="checkbox"
disabled={!selectable}
checked={selected}
onChange={onChangeCheckbox}
disabled={disabled}
checked={checked}
onChange={e => onCheckedItem(e.target.checked, position.id)}
/>
<label htmlFor={`checkbox-item-${position.id}`} />
<DoubleLogo {...doubleLogo} size={24} />
<DoubleLogo left={tokenA.logoURI} right={tokenB.logoURI} size={24} />
<Tooltip
placement="top"
FloatingContent={<TooltipContent selectable={!selectable} />}
FloatingContent={<TooltipContent position={position} disabled={disabled} />}
>
<span className="token-id">GNS/GNOT</span>
<span className="token-id">{`${tokenA.symbol}/${tokenB.symbol}`}</span>
</Tooltip>
<Badge text="0.3%" type={BADGE_TYPE.DARK_DEFAULT} />
<Badge text={feeStr} type={BADGE_TYPE.DARK_DEFAULT} />
</div>
<span className="liquidity-value-fake" ref={liquidityRef}>${position.liquidity.toLocaleString()}</span>
<span className="liquidity-value" >${!checkWidth ? convertLiquidity(position.liquidity.toString()) : position.liquidity.toLocaleString()}</span>
<span className="liquidity-value" >{liquidityUSD}</span>
</RemoveLiquiditySelectListItemWrapper>
);
};
Expand Down
Original file line number Diff line number Diff line change
@@ -1,53 +1,57 @@
import React, { useCallback } from "react";
import React from "react";
import { RemoveLiquiditySelectListWrapper } from "./RemoveLiquiditySelectList.styles";
import RemoveLiquiditySelectListItem from "../remove-liquidity-select-list-item/RemoveLiquiditySelectListItem";
import { PoolPositionModel } from "@models/position/pool-position-model";

interface RemoveLiquiditySelectListProps {
selectedAll: boolean;
positions: PoolPositionModel[];
selectedIds: string[];
select: (id: string) => void;
selectAll: () => void;
width: number;
stakedPositions: PoolPositionModel[];
unstakedPositions: PoolPositionModel[];
checkedList: string[];
onCheckedItem: (checked: boolean, path: string) => void;
onCheckedAll: (checked: boolean) => void;
checkedAll: boolean;
}

const RemoveLiquiditySelectList: React.FC<RemoveLiquiditySelectListProps> = ({
selectedAll,
positions,
selectedIds,
select,
selectAll,
width,
stakedPositions,
unstakedPositions,
checkedList,
onCheckedItem,
onCheckedAll,
checkedAll,
}) => {

const isSelectLiquidity = useCallback((position: PoolPositionModel) => {
return selectedIds.findIndex(id => id === position.id) > -1;
}, [selectedIds]);

return (
<RemoveLiquiditySelectListWrapper>
<div className="checked-all-wrap">
<div className="wrapper-check-label">
<input
id="checkbox-all"
type="checkbox"
checked={selectedAll}
onChange={selectAll}
checked={checkedAll}
onChange={e => onCheckedAll(e.target.checked)}
/>
<label htmlFor="checkbox-all" />
<span className="custom-label">Select All</span>
</div>
<span>Liquidity</span>
</div>
<ul>
{positions.map((position, index) => (
{unstakedPositions.map((position, index) => (
<RemoveLiquiditySelectListItem
position={position}
checkedList={checkedList}
onCheckedItem={onCheckedItem}
key={index}
/>
))}
{stakedPositions.map((position, index) => (
<RemoveLiquiditySelectListItem
position={position}
selected={isSelectLiquidity(position)}
select={select}
width={width}
checkedList={checkedList}
onCheckedItem={onCheckedItem}
key={index}
disabled
/>
))}
</ul>
Expand Down
Loading