import React, { useRef, useEffect, useState } from 'react' import styled from '@emotion/styled' import xw from 'xwind' import useInterval from '../hooks/useInterval' import usePrevious from '../hooks/usePrevious' import { isEqual, getDecimalCount } from '../utils/' import { ArrowUpIcon, ArrowDownIcon } from '@heroicons/react/solid' import useMarkPrice from '../hooks/useMarkPrice' import useOrderbook from '../hooks/useOrderbook' import useMarkets from '../hooks/useMarkets' const Line = styled.div` text-align: ${(props) => (props.invert ? 'left' : 'right')}; float: ${(props) => (props.invert ? 'left' : 'right')}; height: 100%; ${(props) => props['data-width'] && `width: ${props['data-width']};`} ${(props) => props['data-bgcolor'] && `background-color: ${props['data-bgcolor']};`} ` // const Price = styled.div` // position: absolute; // ${(props) => (props.invert ? `left: 5px;` : `right: 15px;`)} // ${(props) => props['data-color'] && `color: ${props['data-color']};`} // ` export default function Orderbook({ depth = 7 }) { // TODO remove smallScreen const smallScreen = false const markPrice = useMarkPrice() const [orderbook] = useOrderbook() const { baseCurrency, quoteCurrency } = useMarkets() 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) const bidsToDisplay = getCumulativeOrderbookSide(bids, totalSize, false) const asksToDisplay = getCumulativeOrderbookSide(asks, totalSize, 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]) function getCumulativeOrderbookSide(orders, totalSize, 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 } return ( <>
Orderbook
<>
Size ({baseCurrency})
Price ({quoteCurrency})
{orderbookData?.asks.map(({ price, size, sizePercent }) => ( alert(`price ${price}`)} onSizeClick={() => alert(`size ${size}`)} /> ))} {orderbookData?.bids.map(({ price, size, sizePercent }) => ( alert(`price ${price}`)} onSizeClick={() => alert(`size ${size}`)} /> ))} ) } const OrderbookRow = React.memo( ({ side, price, size, sizePercent, onSizeClick, onPriceClick, invert }) => { const element = useRef(null) const { market } = useMarkets() 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 (
{invert ? ( <>
{formattedPrice}
{formattedSize}
) : ( <>
{formattedSize}
{formattedPrice}
)}
) }, (prevProps, nextProps) => isEqual(prevProps, nextProps, ['price', 'size', 'sizePercent']) ) const MarkPriceComponent = React.memo<{ markPrice: number }>( ({ markPrice }) => { const { market } = useMarkets() const previousMarkPrice: number = usePrevious(markPrice) const markPriceColor = markPrice > previousMarkPrice ? '#afd803' : markPrice < previousMarkPrice ? '#E54033' : 'white' const formattedMarkPrice = markPrice && market?.tickSize && markPrice.toFixed(getDecimalCount(market.tickSize)) return (
{markPrice > previousMarkPrice && ( )} {markPrice < previousMarkPrice && ( )} {formattedMarkPrice || '----'}
) }, (prevProps, nextProps) => isEqual(prevProps, nextProps, ['markPrice']) )