mango-v4-ui/components/swap/SwapTriggerOrders.tsx

546 lines
18 KiB
TypeScript
Raw Normal View History

2023-08-08 20:40:26 -07:00
import { IconButton, LinkButton } from '@components/shared/Button'
import ConnectEmptyState from '@components/shared/ConnectEmptyState'
import {
SortableColumnHeader,
Table,
Td,
Th,
TrBody,
TrHead,
} from '@components/shared/TableElements'
2023-08-16 20:57:39 -07:00
import {
ChevronDownIcon,
NoSymbolIcon,
TrashIcon,
} from '@heroicons/react/20/solid'
import { BN } from '@project-serum/anchor'
import { useWallet } from '@solana/wallet-adapter-react'
import mangoStore from '@store/mangoStore'
import useMangoAccount from 'hooks/useMangoAccount'
import useMangoGroup from 'hooks/useMangoGroup'
import { useSortableData } from 'hooks/useSortableData'
2023-08-16 20:57:39 -07:00
import { useViewport } from 'hooks/useViewport'
import { useCallback, useMemo, useState } from 'react'
import { useTranslation } from 'react-i18next'
import { notify } from 'utils/notifications'
2023-09-10 05:00:06 -07:00
import { floorToDecimal, formatNumericValue } from 'utils/numbers'
2023-08-16 20:57:39 -07:00
import { breakpoints } from 'utils/theme'
import * as sentry from '@sentry/nextjs'
import { isMangoError } from 'types'
import Loading from '@components/shared/Loading'
2023-08-08 19:53:59 -07:00
import SideBadge from '@components/shared/SideBadge'
2023-08-16 20:57:39 -07:00
import { Disclosure, Transition } from '@headlessui/react'
import SheenLoader from '@components/shared/SheenLoader'
import { formatTokenSymbol } from 'utils/tokens'
2023-11-05 20:00:41 -08:00
export const handleCancelTriggerOrder = async (
id: BN,
setCancelId?: (id: string) => void,
) => {
try {
const client = mangoStore.getState().client
const group = mangoStore.getState().group
const actions = mangoStore.getState().actions
const mangoAccount = mangoStore.getState().mangoAccount.current
if (!mangoAccount || !group) return
if (setCancelId) {
setCancelId(id.toString())
}
try {
const { signature: tx, slot } = await client.tokenConditionalSwapCancel(
group,
mangoAccount,
id,
)
notify({
title: 'Transaction confirmed',
type: 'success',
txid: tx,
noSound: true,
})
actions.fetchGroup()
await actions.reloadMangoAccount(slot)
} catch (e) {
console.error('failed to cancel swap order', e)
sentry.captureException(e)
if (isMangoError(e)) {
notify({
title: 'Transaction failed',
description: e.message,
txid: e?.txid,
type: 'error',
})
}
}
} catch (e) {
console.error('failed to cancel trigger order', e)
} finally {
if (setCancelId) {
setCancelId('')
}
}
}
2023-09-12 18:10:21 -07:00
export const handleCancelAll = async (
setCancelId: (id: '' | 'all') => void,
) => {
try {
const client = mangoStore.getState().client
const group = mangoStore.getState().group
const actions = mangoStore.getState().actions
const mangoAccount = mangoStore.getState().mangoAccount.current
if (!mangoAccount || !group) return
setCancelId('all')
try {
const { signature: tx, slot } =
await client.tokenConditionalSwapCancelAll(group, mangoAccount)
notify({
title: 'Transaction confirmed',
type: 'success',
txid: tx,
noSound: true,
})
actions.fetchGroup()
await actions.reloadMangoAccount(slot)
} catch (e) {
console.error('failed to cancel trigger orders', e)
sentry.captureException(e)
if (isMangoError(e)) {
notify({
title: 'Transaction failed',
description: e.message,
txid: e?.txid,
type: 'error',
})
}
}
} catch (e) {
console.error('failed to cancel swap order', e)
} finally {
setCancelId('')
}
}
const SwapOrders = () => {
const { t } = useTranslation(['common', 'swap', 'trade'])
2023-08-16 20:57:39 -07:00
const { width } = useViewport()
const showTableView = width ? width > breakpoints.md : false
const { mangoAccount, mangoAccountAddress } = useMangoAccount()
const { group } = useMangoGroup()
const { connected } = useWallet()
const [cancelId, setCancelId] = useState('')
const orders = useMemo(() => {
if (!mangoAccount) return []
return mangoAccount.tokenConditionalSwaps.filter((tcs) => tcs.hasData)
}, [mangoAccount])
const formattedTableData = useCallback(() => {
if (!group) return []
const formatted = []
for (const order of orders) {
const buyBank = group.getFirstBankByTokenIndex(order.buyTokenIndex)
const sellBank = group.getFirstBankByTokenIndex(order.sellTokenIndex)
2023-08-08 19:53:59 -07:00
const maxBuy = floorToDecimal(
order.getMaxBuyUi(group),
buyBank.mintDecimals,
).toNumber()
const maxSell = floorToDecimal(
order.getMaxSellUi(group),
sellBank.mintDecimals,
).toNumber()
2023-08-08 19:53:59 -07:00
let size
let side
if (maxBuy === 0 || maxBuy > maxSell) {
size = maxSell
side = 'sell'
} else {
size = maxBuy
side = 'buy'
}
2023-09-10 05:00:06 -07:00
const formattedBuyTokenName = formatTokenSymbol(buyBank.name)
const formattedSellTokenName = formatTokenSymbol(sellBank.name)
const pair =
side === 'sell'
2023-09-10 05:00:06 -07:00
? `${formattedSellTokenName}/${formattedBuyTokenName}`
: `${formattedBuyTokenName}/${formattedSellTokenName}`
2023-08-08 18:03:54 -07:00
const triggerPrice = order.getThresholdPriceUi(group)
const pricePremium = order.getPricePremium()
2023-07-31 05:25:46 -07:00
const filled = order.getSoldUi(group)
2023-08-13 05:47:25 -07:00
const currentPrice = order.getCurrentPairPriceUi(group)
const sellTokenPerBuyToken = !!Object.prototype.hasOwnProperty.call(
order.priceDisplayStyle,
'sellTokenPerBuyToken',
)
2023-09-10 05:00:06 -07:00
const baseTokenName =
side === 'buy' ? formattedBuyTokenName : formattedSellTokenName
const quoteTokenName = !sellTokenPerBuyToken
? formattedBuyTokenName
: formattedSellTokenName
const quoteDecimals = !sellTokenPerBuyToken
? buyBank.mintDecimals
: sellBank.mintDecimals
const triggerDirection = triggerPrice < currentPrice ? '<=' : '>='
const data = {
...order,
2023-09-10 05:00:06 -07:00
baseTokenName,
2023-08-01 22:32:20 -07:00
currentPrice,
2023-09-10 05:00:06 -07:00
fee: pricePremium,
filled,
2023-07-31 05:25:46 -07:00
pair,
2023-09-10 05:00:06 -07:00
quoteDecimals,
quoteTokenName,
2023-08-08 19:53:59 -07:00
side,
size,
triggerDirection,
2023-09-10 05:00:06 -07:00
triggerPrice,
}
formatted.push(data)
}
return formatted
}, [group, orders])
const {
items: tableData,
requestSort,
sortConfig,
} = useSortableData(formattedTableData())
return orders.length ? (
2023-08-16 20:57:39 -07:00
showTableView ? (
<Table>
<thead>
<TrHead>
<Th className="text-left">
<SortableColumnHeader
2023-08-16 20:57:39 -07:00
sortKey="pair"
sort={() => requestSort('pair')}
sortConfig={sortConfig}
2023-08-16 20:57:39 -07:00
title={t('swap:pair')}
/>
2023-08-16 20:57:39 -07:00
</Th>
<Th>
<div className="flex justify-end">
<SortableColumnHeader
sortKey="side"
sort={() => requestSort('side')}
sortConfig={sortConfig}
title={t('trade:side')}
/>
</div>
</Th>
<Th>
<div className="flex justify-end">
<SortableColumnHeader
sortKey="size"
sort={() => requestSort('size')}
sortConfig={sortConfig}
title={t('trade:size')}
/>
</div>
</Th>
<Th>
<div className="flex justify-end">
<SortableColumnHeader
sortKey="filled"
sort={() => requestSort('filled')}
sortConfig={sortConfig}
title={t('trade:filled')}
/>
</div>
</Th>
<Th>
<div className="flex justify-end">
<SortableColumnHeader
sortKey="currentPrice"
sort={() => requestSort('currentPrice')}
sortConfig={sortConfig}
title={t('trade:current-price')}
/>
</div>
</Th>
<Th>
<div className="flex justify-end">
<SortableColumnHeader
sortKey="triggerPrice"
sort={() => requestSort('triggerPrice')}
sortConfig={sortConfig}
title={t('trade:trigger-price')}
/>
</div>
</Th>
<Th>
<div className="flex justify-end">
<SortableColumnHeader
sortKey="fee"
sort={() => requestSort('fee')}
sortConfig={sortConfig}
title={t('trade:est-slippage')}
/>
</div>
</Th>
<Th>
<div className="flex justify-end">
2023-09-12 18:10:21 -07:00
<LinkButton onClick={() => handleCancelAll(setCancelId)}>
2023-08-16 20:57:39 -07:00
{t('trade:cancel-all')}
</LinkButton>
</div>
</Th>
</TrHead>
</thead>
<tbody>
{tableData.map((data, i) => {
const {
2023-09-10 05:00:06 -07:00
baseTokenName,
2023-08-16 20:57:39 -07:00
currentPrice,
fee,
2023-09-10 05:00:06 -07:00
filled,
2023-08-16 20:57:39 -07:00
pair,
2023-09-10 05:00:06 -07:00
quoteDecimals,
quoteTokenName,
2023-08-16 20:57:39 -07:00
side,
size,
triggerDirection,
2023-09-10 05:00:06 -07:00
triggerPrice,
2023-08-16 20:57:39 -07:00
} = data
return (
<TrBody key={i} className="text-sm">
<Td>{pair}</Td>
<Td>
<div className="flex justify-end">
<SideBadge side={side} />
</div>
</Td>
<Td>
<p className="text-right">
{size}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{baseTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</Td>
<Td>
<p className="text-right">
{filled}/{size}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{baseTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</Td>
<Td>
<p className="text-right">
2023-09-10 05:00:06 -07:00
{formatNumericValue(currentPrice, quoteDecimals)}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{quoteTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</Td>
<Td>
<p className="text-right">
<span className="font-body text-th-fgd-4">
{triggerDirection}{' '}
</span>
2023-09-10 05:00:06 -07:00
{formatNumericValue(triggerPrice, quoteDecimals)}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{quoteTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</Td>
<Td>
<p className="text-right">{fee.toFixed(2)}%</p>
</Td>
<Td className="flex justify-end">
<IconButton
disabled={
cancelId === data.id.toString() || cancelId === 'all'
}
2023-11-05 20:00:41 -08:00
onClick={() =>
handleCancelTriggerOrder(data.id, setCancelId)
}
2023-08-16 20:57:39 -07:00
size="small"
>
{cancelId === data.id.toString() || cancelId === 'all' ? (
<Loading />
) : (
<TrashIcon className="h-4 w-4" />
)}
</IconButton>
</Td>
</TrBody>
)
})}
</tbody>
</Table>
) : (
<div className="border-b border-th-bkg-3">
{tableData.map((data, i) => {
const {
2023-09-10 05:00:06 -07:00
baseTokenName,
2023-08-01 22:32:20 -07:00
currentPrice,
fee,
2023-09-10 05:00:06 -07:00
filled,
2023-07-31 05:25:46 -07:00
pair,
2023-09-10 05:00:06 -07:00
quoteDecimals,
quoteTokenName,
2023-08-08 19:53:59 -07:00
side,
size,
2023-09-08 05:06:58 -07:00
triggerDirection,
2023-09-10 05:00:06 -07:00
triggerPrice,
} = data
2023-08-08 19:53:59 -07:00
return (
2023-08-16 20:57:39 -07:00
<Disclosure key={i}>
{({ open }) => (
<>
<Disclosure.Button
className={`w-full border-t border-th-bkg-3 p-4 text-left focus:outline-none ${
i === 0 ? 'border-t-0' : ''
}`}
>
<div className="flex items-center justify-between">
<div>
<span className="mr-1 whitespace-nowrap">{pair}</span>
<SideBadge side={side} />
<p className="font-mono text-th-fgd-2">
{size}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{baseTokenName}
2023-08-16 20:57:39 -07:00
</span>
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' at '}
</span>
2023-09-10 05:00:06 -07:00
{formatNumericValue(triggerPrice, quoteDecimals)}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{quoteTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</div>
<ChevronDownIcon
className={`${
open ? 'rotate-180' : 'rotate-360'
} h-6 w-6 flex-shrink-0 text-th-fgd-3`}
/>
</div>
</Disclosure.Button>
<Transition
enter="transition ease-in duration-200"
enterFrom="opacity-0"
enterTo="opacity-100"
>
<Disclosure.Panel>
<div className="mx-4 grid grid-cols-2 gap-4 border-t border-th-bkg-3 pb-4 pt-4">
2023-08-16 20:57:39 -07:00
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('trade:size')}
</p>
<p className="font-mono text-th-fgd-1">
{size}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{baseTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('trade:filled')}
</p>
<p className="font-mono text-th-fgd-1">
{filled}/{size}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{baseTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('trade:current-price')}
</p>
<p className="font-mono text-th-fgd-1">
2023-09-10 05:00:06 -07:00
{formatNumericValue(currentPrice, quoteDecimals)}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{quoteTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('trade:trigger-price')}
</p>
<p className="font-mono text-th-fgd-1">
2023-09-08 05:06:58 -07:00
<span className="font-body text-th-fgd-4">
{triggerDirection}{' '}
</span>
2023-09-10 05:00:06 -07:00
{formatNumericValue(triggerPrice, quoteDecimals)}
<span className="font-body text-th-fgd-3">
2023-08-16 20:57:39 -07:00
{' '}
2023-09-10 05:00:06 -07:00
{quoteTokenName}
2023-08-16 20:57:39 -07:00
</span>
</p>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('trade:est-slippage')}
</p>
<p className="font-mono text-th-fgd-1">
{fee.toFixed(2)}%
</p>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">{t('cancel')}</p>
2023-11-05 20:00:41 -08:00
<LinkButton
onClick={() =>
handleCancelTriggerOrder(data.id, setCancelId)
}
>
2023-08-16 20:57:39 -07:00
{cancelId === data.id.toString() ? (
<SheenLoader className="mt-1">
<div className="h-3.5 w-20 bg-th-bkg-2" />
</SheenLoader>
) : (
t('trade:cancel-order')
)}
</LinkButton>
</div>
</div>
</Disclosure.Panel>
</Transition>
</>
)}
</Disclosure>
)
})}
2023-08-16 20:57:39 -07:00
</div>
)
) : mangoAccountAddress || connected ? (
2023-10-17 04:28:43 -07:00
<div className="flex flex-1 flex-col items-center justify-center">
<div className="flex flex-col items-center p-8">
<NoSymbolIcon className="mb-2 h-6 w-6 text-th-fgd-4" />
<p>{t('trade:no-trigger-orders')}</p>
</div>
</div>
) : (
2023-10-17 04:28:43 -07:00
<div className="flex flex-1 flex-col items-center justify-center p-8">
<ConnectEmptyState text={t('trade:connect-trigger-orders')} />
</div>
)
}
export default SwapOrders