import { PerpMarket, PerpOrder, PerpOrderType, Serum3Market, Serum3OrderType, Serum3SelfTradeBehavior, Serum3Side, } from '@blockworks-foundation/mango-v4' import Input from '@components/forms/Input' import { IconButton } from '@components/shared/Button' import Loading from '@components/shared/Loading' import SideBadge from '@components/shared/SideBadge' import { Table, Td, Th, TrBody, TrHead } from '@components/shared/TableElements' import Tooltip from '@components/shared/Tooltip' import { CheckIcon, NoSymbolIcon, PencilIcon, TrashIcon, XMarkIcon, } from '@heroicons/react/20/solid' import { Order } from '@project-serum/serum/lib/market' import { useWallet } from '@solana/wallet-adapter-react' import { PublicKey } from '@solana/web3.js' import mangoStore from '@store/mangoStore' import useMangoAccount from 'hooks/useMangoAccount' import { useViewport } from 'hooks/useViewport' import { useTranslation } from 'next-i18next' import { ChangeEvent, useCallback, useState } from 'react' import { notify } from 'utils/notifications' import { formatFixedDecimals, getDecimalCount } from 'utils/numbers' import { breakpoints } from 'utils/theme' import TableMarketName from './TableMarketName' const OpenOrders = () => { const { t } = useTranslation(['common', 'trade']) const openOrders = mangoStore((s) => s.mangoAccount.openOrders) const [cancelId, setCancelId] = useState('') const [modifyOrderId, setModifyOrderId] = useState( undefined ) const [loadingModifyOrder, setLoadingModifyOrder] = useState(false) const [modifiedOrderSize, setModifiedOrderSize] = useState('') const [modifiedOrderPrice, setModifiedOrderPrice] = useState('') const { width } = useViewport() const showTableView = width ? width > breakpoints.md : false const { mangoAccount } = useMangoAccount() const { connected } = useWallet() const findSerum3MarketPkInOpenOrders = (o: Order): string | undefined => { const openOrders = mangoStore.getState().mangoAccount.openOrders let foundedMarketPk: string | undefined = undefined for (const [marketPk, orders] of Object.entries(openOrders)) { for (const order of orders) { if (order.orderId.eq(o.orderId)) { foundedMarketPk = marketPk break } } if (foundedMarketPk) { break } } return foundedMarketPk } const handleCancelSerumOrder = useCallback( async (o: Order) => { const client = mangoStore.getState().client const group = mangoStore.getState().group const mangoAccount = mangoStore.getState().mangoAccount.current const actions = mangoStore.getState().actions if (!group || !mangoAccount) return const marketPk = findSerum3MarketPkInOpenOrders(o) if (!marketPk) return const market = group.getSerum3MarketByExternalMarket( new PublicKey(marketPk) ) setCancelId(o.orderId.toString()) try { const tx = await client.serum3CancelOrder( group, mangoAccount, market!.serumMarketExternal, o.side === 'buy' ? Serum3Side.bid : Serum3Side.ask, o.orderId ) actions.fetchOpenOrders() notify({ type: 'success', title: 'Transaction successful', txid: tx, }) } catch (e: any) { console.error('Error canceling', e) notify({ title: t('trade:cancel-order-error'), description: e.message, txid: e.txid, type: 'error', }) } finally { setCancelId('') } }, [t, openOrders] ) const modifyOrder = useCallback( async (o: PerpOrder | Order) => { const client = mangoStore.getState().client const group = mangoStore.getState().group const mangoAccount = mangoStore.getState().mangoAccount.current const actions = mangoStore.getState().actions const baseSize = modifiedOrderSize ? Number(modifiedOrderSize) : o.size const price = modifiedOrderPrice ? Number(modifiedOrderPrice) : o.price if (!group || !mangoAccount) return setLoadingModifyOrder(true) try { let tx = '' if (o instanceof PerpOrder) { tx = await client.modifyPerpOrder( group, mangoAccount, o.perpMarketIndex, o.orderId, o.side, price, Math.abs(baseSize), undefined, // maxQuoteQuantity Date.now(), PerpOrderType.limit, undefined, undefined ) } else { const marketPk = findSerum3MarketPkInOpenOrders(o) if (!marketPk) return const market = group.getSerum3MarketByExternalMarket( new PublicKey(marketPk) ) tx = await client.modifySerum3Order( group, o.orderId, mangoAccount, market.serumMarketExternal, o.side === 'buy' ? Serum3Side.bid : Serum3Side.ask, price, baseSize, Serum3SelfTradeBehavior.decrementTake, Serum3OrderType.limit, Date.now(), 10 ) } actions.fetchOpenOrders() notify({ type: 'success', title: 'Transaction successful', txid: tx, }) } catch (e: any) { console.error('Error canceling', e) notify({ title: 'Unable to modify order', description: e.message, txid: e.txid, type: 'error', }) } finally { cancelEditOrderForm() } }, [t, modifiedOrderSize, modifiedOrderPrice] ) const handleCancelPerpOrder = useCallback( async (o: PerpOrder) => { const client = mangoStore.getState().client const group = mangoStore.getState().group const mangoAccount = mangoStore.getState().mangoAccount.current const actions = mangoStore.getState().actions if (!group || !mangoAccount) return setCancelId(o.orderId.toString()) try { const tx = await client.perpCancelOrder( group, mangoAccount, o.perpMarketIndex, o.orderId ) actions.fetchOpenOrders() notify({ type: 'success', title: 'Transaction successful', txid: tx, }) } catch (e: any) { console.error('Error canceling', e) notify({ title: t('trade:cancel-order-error'), description: e.message, txid: e.txid, type: 'error', }) } finally { setCancelId('') } }, [t] ) const showEditOrderForm = (order: Order | PerpOrder, tickSize: number) => { setModifyOrderId(order.orderId.toString()) setModifiedOrderSize(order.size.toString()) setModifiedOrderPrice(order.price.toFixed(getDecimalCount(tickSize))) } const cancelEditOrderForm = () => { setModifyOrderId(undefined) setLoadingModifyOrder(false) setModifiedOrderSize('') setModifiedOrderPrice('') } return mangoAccount && Object.values(openOrders).flat().length ? ( showTableView ? ( {connected ? : null} {Object.entries(openOrders) .map(([marketPk, orders]) => { return orders.map((o) => { const group = mangoStore.getState().group! let market: PerpMarket | Serum3Market let tickSize: number let minOrderSize: number let quoteSymbol if (o instanceof PerpOrder) { market = group.getPerpMarketByMarketIndex(o.perpMarketIndex) quoteSymbol = group.getFirstBankByTokenIndex( market.settleTokenIndex ).name tickSize = market.tickSize minOrderSize = market.minOrderSize } else { market = group.getSerum3MarketByExternalMarket( new PublicKey(marketPk) ) quoteSymbol = group.getFirstBankByTokenIndex( market!.quoteTokenIndex ).name const serumMarket = group.getSerum3ExternalMarket( market.serumMarketExternal ) tickSize = serumMarket.tickSize minOrderSize = serumMarket.minOrderSize } return ( {modifyOrderId !== o.orderId.toString() ? ( <> ) : ( <> )} {connected ? ( ) : null} ) }) }) .flat()}
{t('market')} {t('trade:side')} {t('trade:size')} {t('price')} {t('value')}
{o.size.toLocaleString(undefined, { maximumFractionDigits: getDecimalCount(minOrderSize), })} {o.price.toLocaleString(undefined, { minimumFractionDigits: getDecimalCount(tickSize), maximumFractionDigits: getDecimalCount(tickSize), })}{' '} {quoteSymbol} ) => setModifiedOrderSize(e.target.value) } /> ) => setModifiedOrderPrice(e.target.value) } /> {formatFixedDecimals(o.size * o.price, true, true)}
{modifyOrderId !== o.orderId.toString() ? ( <> showEditOrderForm(o, tickSize)} size="small" > o instanceof PerpOrder ? handleCancelPerpOrder(o) : handleCancelSerumOrder(o) } size="small" > {cancelId === o.orderId.toString() ? ( ) : ( )} ) : ( <> modifyOrder(o)} size="small" > {loadingModifyOrder ? ( ) : ( )} )}
) : (
{Object.entries(openOrders).map(([marketPk, orders]) => { return orders.map((o) => { const group = mangoStore.getState().group! let market: PerpMarket | Serum3Market let tickSize: number let minOrderSize: number let quoteSymbol: string let baseSymbol: string if (o instanceof PerpOrder) { market = group.getPerpMarketByMarketIndex(o.perpMarketIndex) baseSymbol = market.name.split('-')[0] quoteSymbol = group.getFirstBankByTokenIndex( market.settleTokenIndex ).name tickSize = market.tickSize minOrderSize = market.minOrderSize } else { market = group.getSerum3MarketByExternalMarket( new PublicKey(marketPk) ) baseSymbol = market.name.split('/')[0] quoteSymbol = group.getFirstBankByTokenIndex( market!.quoteTokenIndex ).name const serumMarket = group.getSerum3ExternalMarket( market.serumMarketExternal ) tickSize = serumMarket.tickSize minOrderSize = serumMarket.minOrderSize } return (
{modifyOrderId !== o.orderId.toString() ? (

{o.size.toLocaleString(undefined, { maximumFractionDigits: getDecimalCount(minOrderSize), })} {' '} {baseSymbol} {' for '} {o.price.toLocaleString(undefined, { minimumFractionDigits: getDecimalCount(tickSize), maximumFractionDigits: getDecimalCount(tickSize), })} {' '} {quoteSymbol}

) : (

{t('trade:size')}

) => setModifiedOrderSize(e.target.value) } />

{t('price')}

) => setModifiedOrderPrice(e.target.value) } />
)}
{connected ? (
{modifyOrderId !== o.orderId.toString() ? ( <> showEditOrderForm(o, tickSize)} > o instanceof PerpOrder ? handleCancelPerpOrder(o) : handleCancelSerumOrder(o) } > {cancelId === o.orderId.toString() ? ( ) : ( )} ) : ( <> modifyOrder(o)}> {loadingModifyOrder ? ( ) : ( )} )}
) : null}
) }) })}
) ) : (

{t('trade:no-orders')}

) } export default OpenOrders