mango-ui-v3/components/Orderbook.tsx

256 lines
7.2 KiB
TypeScript
Raw Normal View History

import React, { useRef, useEffect, useState } from 'react'
import styled from '@emotion/styled'
import useInterval from '../hooks/useInterval'
import usePrevious from '../hooks/usePrevious'
import { isEqual, getDecimalCount } from '../utils/'
2021-04-02 11:26:21 -07:00
import { ArrowUpIcon, ArrowDownIcon } from '@heroicons/react/solid'
import useMarkPrice from '../hooks/useMarkPrice'
import useOrderbook from '../hooks/useOrderbook'
2021-04-06 15:11:42 -07:00
import useMarket from '../hooks/useMarket'
2021-04-05 13:48:24 -07:00
import { ElementTitle } from './styles'
2021-04-13 16:41:04 -07:00
import useMangoStore from '../stores/useMangoStore'
const Line = styled.div<any>`
text-align: ${(props) => (props.invert ? 'left' : 'right')};
float: ${(props) => (props.invert ? 'left' : 'right')};
height: 100%;
2021-04-16 04:50:56 -07:00
filter: opacity(70%);
${(props) => props['data-width'] && `width: ${props['data-width']};`}
`
2021-04-07 08:44:22 -07:00
function getCumulativeOrderbookSide(
orders,
totalSize,
depth,
backwards = false
) {
let cumulative = orders
.slice(0, depth)
.reduce((cumulative, [price, size], i) => {
const cumulativeSize = (cumulative[i - 1]?.cumulativeSize || 0) + size
cumulative.push({
price,
size,
cumulativeSize,
sizePercent: Math.round((cumulativeSize / (totalSize || 1)) * 100),
})
return cumulative
}, [])
if (backwards) {
cumulative = cumulative.reverse()
}
return cumulative
}
export default function Orderbook({ depth = 7 }) {
const markPrice = useMarkPrice()
const [orderbook] = useOrderbook()
2021-04-06 15:11:42 -07:00
const { baseCurrency, quoteCurrency } = useMarket()
2021-04-13 16:41:04 -07:00
const setMangoStore = useMangoStore((s) => s.set)
const currentOrderbookData = useRef(null)
const lastOrderbookData = useRef(null)
const [orderbookData, setOrderbookData] = useState(null)
useInterval(() => {
if (
!currentOrderbookData.current ||
JSON.stringify(currentOrderbookData.current) !==
JSON.stringify(lastOrderbookData.current)
) {
const bids = orderbook?.bids || []
const asks = orderbook?.asks || []
const sum = (total, [, size], index) =>
index < depth ? total + size : total
const totalSize = bids.reduce(sum, 0) + asks.reduce(sum, 0)
2021-04-07 08:44:22 -07:00
const bidsToDisplay = getCumulativeOrderbookSide(
bids,
totalSize,
depth,
false
)
const asksToDisplay = getCumulativeOrderbookSide(
asks,
totalSize,
depth,
true
)
currentOrderbookData.current = {
bids: orderbook?.bids,
asks: orderbook?.asks,
}
setOrderbookData({ bids: bidsToDisplay, asks: asksToDisplay })
}
}, 250)
useEffect(() => {
lastOrderbookData.current = {
bids: orderbook?.bids,
asks: orderbook?.asks,
}
}, [orderbook])
2021-04-13 16:41:04 -07:00
const handlePriceClick = (price) => {
2021-04-15 14:36:55 -07:00
console.log('price click')
2021-04-13 16:41:04 -07:00
setMangoStore((state) => {
state.tradeForm.price = price
})
}
const handleSizeClick = (size) => {
2021-04-15 14:36:55 -07:00
console.log('size click')
2021-04-13 16:41:04 -07:00
setMangoStore((state) => {
state.tradeForm.baseSize = size
})
}
2021-04-02 11:26:21 -07:00
return (
<>
2021-04-05 13:48:24 -07:00
<ElementTitle>Orderbook</ElementTitle>
<div className={`text-th-fgd-4 flex justify-between mb-2`}>
<div className={`text-left`}>Size ({baseCurrency})</div>
<div className={`text-right`}>Price ({quoteCurrency})</div>
2021-04-05 13:48:24 -07:00
</div>
{orderbookData?.asks.map(({ price, size, sizePercent }) => (
<OrderbookRow
key={price + ''}
price={price}
size={size}
side={'sell'}
sizePercent={sizePercent}
2021-04-13 16:41:04 -07:00
onPriceClick={() => handlePriceClick(price)}
onSizeClick={() => handleSizeClick(size)}
2021-04-05 13:48:24 -07:00
/>
))}
<MarkPriceComponent markPrice={markPrice} />
{orderbookData?.bids.map(({ price, size, sizePercent }) => (
<OrderbookRow
key={price + ''}
price={price}
size={size}
side={'buy'}
sizePercent={sizePercent}
2021-04-13 16:41:04 -07:00
onPriceClick={() => handlePriceClick(price)}
onSizeClick={() => handleSizeClick(size)}
2021-04-05 13:48:24 -07:00
/>
))}
2021-04-02 11:26:21 -07:00
</>
)
}
2021-04-02 11:26:21 -07:00
const OrderbookRow = React.memo<any>(
({ side, price, size, sizePercent, onSizeClick, onPriceClick, invert }) => {
2021-04-02 11:26:21 -07:00
const element = useRef(null)
2021-04-06 15:11:42 -07:00
const { market } = useMarket()
useEffect(() => {
!element.current?.classList.contains('flash') &&
element.current?.classList.add('flash')
const id = setTimeout(
() =>
element.current?.classList.contains('flash') &&
element.current?.classList.remove('flash'),
250
)
return () => clearTimeout(id)
}, [price, size])
const formattedSize =
market?.minOrderSize && !isNaN(size)
? Number(size).toFixed(getDecimalCount(market.minOrderSize) + 1)
: size
const formattedPrice =
market?.tickSize && !isNaN(price)
? Number(price).toFixed(getDecimalCount(market.tickSize) + 1)
: price
return (
2021-04-16 04:50:56 -07:00
<div className={`flex text-sm leading-7 justify-between`} ref={element}>
{invert ? (
<>
<div className={`text-left`}>
<Line
invert
data-width={sizePercent + '%'}
side={side}
className={`${side === 'buy' ? `bg-th-green` : `bg-th-red`}`}
/>
2021-04-12 06:32:01 -07:00
<div onClick={onPriceClick}>{formattedPrice}</div>
</div>
2021-04-15 14:36:55 -07:00
<div className={`text-right`} onClick={onSizeClick}>
{formattedSize}
</div>
</>
) : (
<>
2021-04-15 14:36:55 -07:00
<div
className={`text-left flex-1 text-th-fgd-1`}
onClick={onSizeClick}
>
{formattedSize}
</div>
<div className={`text-right relative flex-1`}>
<Line
className={`absolute inset-y-0 right-0 ${
side === 'buy' ? `bg-th-green` : `bg-th-red`
}`}
data-width={sizePercent + '%'}
2021-04-12 06:32:01 -07:00
side={side}
/>
<div
className={`z-30 relative text-th-fgd-1`}
onClick={onPriceClick}
>
{formattedPrice}
2021-04-05 12:03:20 -07:00
</div>
2021-04-02 11:26:21 -07:00
</div>
</>
)}
</div>
)
},
(prevProps, nextProps) =>
isEqual(prevProps, nextProps, ['price', 'size', 'sizePercent'])
)
2021-04-02 11:26:21 -07:00
const MarkPriceComponent = React.memo<{ markPrice: number }>(
({ markPrice }) => {
2021-04-06 15:11:42 -07:00
const { market } = useMarket()
2021-04-02 11:26:21 -07:00
const previousMarkPrice: number = usePrevious(markPrice)
const formattedMarkPrice =
markPrice &&
market?.tickSize &&
markPrice.toFixed(getDecimalCount(market.tickSize))
return (
2021-04-02 11:26:21 -07:00
<div
className={`flex justify-center items-center font-bold p-1 ${
2021-04-12 06:32:01 -07:00
markPrice > previousMarkPrice
? `text-th-green`
2021-04-12 06:32:01 -07:00
: markPrice < previousMarkPrice
? `text-th-red`
: `text-th-fgd-1`
}`}
2021-04-02 11:26:21 -07:00
>
{markPrice > previousMarkPrice && (
2021-04-19 06:45:59 -07:00
<ArrowUpIcon className={`h-4 w-4 mr-1 text-th-green`} />
2021-04-02 11:26:21 -07:00
)}
{markPrice < previousMarkPrice && (
2021-04-19 06:45:59 -07:00
<ArrowDownIcon className={`h-4 w-4 mr-1 text-th-red`} />
2021-04-02 11:26:21 -07:00
)}
{formattedMarkPrice || '----'}
</div>
)
},
(prevProps, nextProps) => isEqual(prevProps, nextProps, ['markPrice'])
)