Update Funding Chart Tooltip (#225)
* 🚧 depth chart * ✨ Fixed types and shortened FundingRateTooltip * 💄 update depth chart color scale to use css var * 💄 use layer-6 instead of text-1 * 🌐 Add localization, fix nits
This commit is contained in:
parent
83bcde00eb
commit
55b2c3fb4a
@ -41,7 +41,7 @@
|
||||
"@cosmjs/tendermint-rpc": "^0.31.0",
|
||||
"@dydxprotocol/v4-abacus": "^1.1.33",
|
||||
"@dydxprotocol/v4-client-js": "^1.0.11",
|
||||
"@dydxprotocol/v4-localization": "^1.1.6",
|
||||
"@dydxprotocol/v4-localization": "^1.1.11",
|
||||
"@ethersproject/providers": "^5.7.2",
|
||||
"@js-joda/core": "^5.5.3",
|
||||
"@radix-ui/react-accordion": "^1.1.2",
|
||||
|
||||
8
pnpm-lock.yaml
generated
8
pnpm-lock.yaml
generated
@ -33,8 +33,8 @@ dependencies:
|
||||
specifier: ^1.0.11
|
||||
version: 1.0.11
|
||||
'@dydxprotocol/v4-localization':
|
||||
specifier: ^1.1.6
|
||||
version: 1.1.6
|
||||
specifier: ^1.1.11
|
||||
version: 1.1.11
|
||||
'@ethersproject/providers':
|
||||
specifier: ^5.7.2
|
||||
version: 5.7.2
|
||||
@ -1023,8 +1023,8 @@ packages:
|
||||
- utf-8-validate
|
||||
dev: false
|
||||
|
||||
/@dydxprotocol/v4-localization@1.1.6:
|
||||
resolution: {integrity: sha512-Bon6NSRU4/FqneAbnP2G28EAPr0hp4LhvAayX61o0O1PGkxnLzAHkXeFppdM0Zn0fOcp1S1MJ+gvz138ZDephQ==}
|
||||
/@dydxprotocol/v4-localization@1.1.11:
|
||||
resolution: {integrity: sha512-ZHnyrWD1bEpvY1tctkcmSV6A5+hQSYBFwyjbiyOqepqDqrv24J5a3hlZU94lDyYf+7Sdk3alOvy9Z7Lpk59nvw==}
|
||||
dev: false
|
||||
|
||||
/@dydxprotocol/v4-proto@0.4.1:
|
||||
|
||||
@ -1,26 +1,33 @@
|
||||
import React, { useContext, useState } from 'react';
|
||||
import React, { PropsWithChildren, useContext, useState } from 'react';
|
||||
|
||||
import { Point } from '@visx/point';
|
||||
import { localPoint } from '@visx/event';
|
||||
import { XYChart, DataContext } from '@visx/xychart';
|
||||
import { XYChart, DataContext, type EventHandlerParams } from '@visx/xychart';
|
||||
import { getScaleBandwidth } from '@/components/visx/getScaleBandwidth';
|
||||
|
||||
export const XYChartWithPointerEvents = ({
|
||||
onPointerMove, onPointerUp, onPointerPressedChange, ...props
|
||||
onPointerMove,
|
||||
onPointerUp,
|
||||
onPointerPressedChange,
|
||||
...props
|
||||
}: {
|
||||
onPointerMove?: (point: Point) => void;
|
||||
onPointerUp?: (point: Point) => void;
|
||||
onPointerMove?: (point: Point | EventHandlerParams<object>) => void;
|
||||
onPointerUp?: (point: Point | EventHandlerParams<object>) => void;
|
||||
onPointerPressedChange?: (isPointerPressed: boolean) => void;
|
||||
} & React.PropsWithChildren<Parameters<typeof XYChart>>) => {
|
||||
} & PropsWithChildren<Parameters<typeof XYChart>[0]>) => {
|
||||
const { xScale, yScale } = useContext(DataContext);
|
||||
|
||||
const [lastPointerMoveEvent, setLastPointerMoveEvent] = useState<React.PointerEvent>();
|
||||
|
||||
const pointerContainerPosition = lastPointerMoveEvent ? localPoint(lastPointerMoveEvent) : null;
|
||||
|
||||
const pointerChartPosition = xScale && yScale && pointerContainerPosition &&
|
||||
const pointerChartPosition =
|
||||
xScale &&
|
||||
yScale &&
|
||||
pointerContainerPosition &&
|
||||
new Point({
|
||||
// @ts-expect-error invert supposedly doesn't exist on AxisScale
|
||||
x: xScale.invert(pointerContainerPosition?.x - getScaleBandwidth(xScale) / 2),
|
||||
// @ts-expect-error invert supposedly doesn't exist on AxisScale
|
||||
y: yScale.invert(pointerContainerPosition?.y - getScaleBandwidth(yScale) / 2),
|
||||
});
|
||||
|
||||
@ -29,15 +36,13 @@ export const XYChartWithPointerEvents = ({
|
||||
{...props}
|
||||
onPointerMove={({ event }) => {
|
||||
setLastPointerMoveEvent(event as React.PointerEvent);
|
||||
if (pointerChartPosition)
|
||||
onPointerMove?.(pointerChartPosition);
|
||||
if (pointerChartPosition) onPointerMove?.(pointerChartPosition);
|
||||
}}
|
||||
onPointerOut={() => setLastPointerMoveEvent(undefined)}
|
||||
onPointerDown={() => onPointerPressedChange?.(true)}
|
||||
onPointerUp={() => {
|
||||
onPointerPressedChange?.(false);
|
||||
if (pointerChartPosition)
|
||||
onPointerUp?.(pointerChartPosition);
|
||||
if (pointerChartPosition) onPointerUp?.(pointerChartPosition);
|
||||
}}
|
||||
>
|
||||
{props.children}
|
||||
|
||||
40
src/constants/charts.ts
Normal file
40
src/constants/charts.ts
Normal file
@ -0,0 +1,40 @@
|
||||
import { OrderSide } from '@dydxprotocol/v4-client-js';
|
||||
import { FundingDirection } from './markets';
|
||||
|
||||
// ------ Depth Chart ------ //
|
||||
export enum DepthChartSeries {
|
||||
Asks = 'Asks',
|
||||
Bids = 'Bids',
|
||||
MidMarket = 'MidMarket',
|
||||
}
|
||||
|
||||
export type DepthChartDatum = {
|
||||
size: number;
|
||||
price: number;
|
||||
depth: number;
|
||||
seriesKey: DepthChartSeries;
|
||||
};
|
||||
|
||||
export type DepthChartPoint = {
|
||||
side: OrderSide;
|
||||
price: number;
|
||||
size: number;
|
||||
};
|
||||
|
||||
export const SERIES_KEY_FOR_ORDER_SIDE = {
|
||||
[OrderSide.BUY]: DepthChartSeries.Bids,
|
||||
[OrderSide.SELL]: DepthChartSeries.Asks,
|
||||
};
|
||||
|
||||
// ------ Funding Chart ------ //
|
||||
export enum FundingRateResolution {
|
||||
OneHour = 'OneHour',
|
||||
EightHour = 'EightHour',
|
||||
Annualized = 'Annualized',
|
||||
}
|
||||
|
||||
export type FundingChartDatum = {
|
||||
time: number;
|
||||
fundingRate: number;
|
||||
direction: FundingDirection;
|
||||
};
|
||||
@ -25,4 +25,5 @@ export const DEFAULT_MARKETID = 'ETH-USD';
|
||||
export enum FundingDirection {
|
||||
ToShort = 'ToShort',
|
||||
ToLong = 'ToLong',
|
||||
None = 'None',
|
||||
}
|
||||
|
||||
44
src/hooks/useOrderbookValues.ts
Normal file
44
src/hooks/useOrderbookValues.ts
Normal file
@ -0,0 +1,44 @@
|
||||
import { useMemo } from 'react';
|
||||
import { shallowEqual, useSelector } from 'react-redux';
|
||||
|
||||
import { DepthChartSeries, DepthChartDatum } from '@/constants/charts';
|
||||
|
||||
import { getCurrentMarketOrderbook } from '@/state/perpetualsSelectors';
|
||||
|
||||
import { MustBigNumber } from '@/lib/numbers';
|
||||
|
||||
export const useOrderbookValuesForDepthChart = () => {
|
||||
const orderbook = useSelector(getCurrentMarketOrderbook, shallowEqual);
|
||||
|
||||
return useMemo(() => {
|
||||
const bids = (orderbook?.bids?.toArray() ?? [])
|
||||
.filter(Boolean)
|
||||
.map((datum) => ({ ...datum, seriesKey: DepthChartSeries.Bids } as DepthChartDatum));
|
||||
|
||||
const asks = (orderbook?.asks?.toArray() ?? [])
|
||||
.filter(Boolean)
|
||||
.map((datum) => ({ ...datum, seriesKey: DepthChartSeries.Asks } as DepthChartDatum));
|
||||
|
||||
const lowestBid = bids[bids.length - 1];
|
||||
const highestBid = bids[0];
|
||||
const lowestAsk = asks[0];
|
||||
const highestAsk = asks[asks.length - 1];
|
||||
|
||||
const midMarketPrice = orderbook?.midPrice;
|
||||
const spread = MustBigNumber(lowestAsk?.price ?? 0).minus(highestBid?.price ?? 0);
|
||||
const spreadPercent = orderbook?.spreadPercent;
|
||||
|
||||
return {
|
||||
bids,
|
||||
asks,
|
||||
lowestBid,
|
||||
highestBid,
|
||||
lowestAsk,
|
||||
highestAsk,
|
||||
midMarketPrice,
|
||||
spread,
|
||||
spreadPercent,
|
||||
orderbook,
|
||||
};
|
||||
}, [orderbook]);
|
||||
};
|
||||
@ -26,7 +26,12 @@ export const calculateFundingRateHistory = createSelector(
|
||||
return data.map(({ effectiveAtMilliseconds, rate }) => ({
|
||||
time: effectiveAtMilliseconds,
|
||||
fundingRate: rate,
|
||||
direction: rate < 0 ? FundingDirection.ToLong : FundingDirection.ToShort,
|
||||
direction:
|
||||
rate === 0
|
||||
? FundingDirection.None
|
||||
: rate < 0
|
||||
? FundingDirection.ToLong
|
||||
: FundingDirection.ToShort,
|
||||
}));
|
||||
}
|
||||
);
|
||||
|
||||
220
src/views/charts/DepthChart/Tooltip.tsx
Normal file
220
src/views/charts/DepthChart/Tooltip.tsx
Normal file
@ -0,0 +1,220 @@
|
||||
import { useMemo } from 'react';
|
||||
import { OrderSide } from '@dydxprotocol/v4-client-js';
|
||||
import type { RenderTooltipParams } from '@visx/xychart/lib/components/Tooltip';
|
||||
import { shallowEqual, useSelector } from 'react-redux';
|
||||
|
||||
import type { Nullable } from '@/constants/abacus';
|
||||
import {
|
||||
DepthChartDatum,
|
||||
DepthChartPoint,
|
||||
DepthChartSeries,
|
||||
SERIES_KEY_FOR_ORDER_SIDE,
|
||||
} from '@/constants/charts';
|
||||
import { STRING_KEYS } from '@/constants/localization';
|
||||
|
||||
import { useStringGetter } from '@/hooks';
|
||||
import { useOrderbookValuesForDepthChart } from '@/hooks/useOrderbookValues';
|
||||
|
||||
import { TooltipContent } from '@/components/visx/TooltipContent';
|
||||
import { Details } from '@/components/Details';
|
||||
import { Output, OutputType } from '@/components/Output';
|
||||
|
||||
import { getCurrentMarketAssetData } from '@/state/assetsSelectors';
|
||||
|
||||
import { MustBigNumber } from '@/lib/numbers';
|
||||
|
||||
type DepthChartTooltipProps = {
|
||||
chartPointAtPointer: DepthChartPoint;
|
||||
isEditingOrder?: boolean;
|
||||
stepSizeDecimals: Nullable<number>;
|
||||
tickSizeDecimals: Nullable<number>;
|
||||
} & Pick<RenderTooltipParams<DepthChartDatum>, 'colorScale' | 'tooltipData'>;
|
||||
|
||||
export const DepthChartTooltipContent = ({
|
||||
chartPointAtPointer,
|
||||
colorScale,
|
||||
isEditingOrder,
|
||||
stepSizeDecimals,
|
||||
tickSizeDecimals,
|
||||
tooltipData,
|
||||
}: DepthChartTooltipProps) => {
|
||||
const { nearestDatum } = tooltipData || {};
|
||||
const stringGetter = useStringGetter();
|
||||
const { spread, spreadPercent, midMarketPrice } = useOrderbookValuesForDepthChart();
|
||||
const { id = '' } = useSelector(getCurrentMarketAssetData, shallowEqual) ?? {};
|
||||
|
||||
const priceImpact = useMemo(() => {
|
||||
if (nearestDatum) {
|
||||
const depthChartSeries = nearestDatum.key as DepthChartSeries;
|
||||
|
||||
return {
|
||||
[DepthChartSeries.Bids]: MustBigNumber(nearestDatum?.datum.price)
|
||||
.minus(chartPointAtPointer.price)
|
||||
.div(nearestDatum?.datum.price),
|
||||
[DepthChartSeries.Asks]: MustBigNumber(chartPointAtPointer.price)
|
||||
.minus(nearestDatum?.datum.price)
|
||||
.div(chartPointAtPointer.price),
|
||||
[DepthChartSeries.MidMarket]: undefined,
|
||||
}[depthChartSeries];
|
||||
}
|
||||
return undefined;
|
||||
}, [nearestDatum, chartPointAtPointer.price]);
|
||||
|
||||
if (!isEditingOrder && !nearestDatum?.datum) return null;
|
||||
|
||||
return (
|
||||
<TooltipContent
|
||||
accentColor={
|
||||
nearestDatum?.key &&
|
||||
colorScale?.(
|
||||
isEditingOrder ? SERIES_KEY_FOR_ORDER_SIDE[chartPointAtPointer.side] : nearestDatum.key
|
||||
)
|
||||
}
|
||||
>
|
||||
<h4>
|
||||
{isEditingOrder
|
||||
? stringGetter({ key: STRING_KEYS.RELEASE_TO_EDIT })
|
||||
: nearestDatum &&
|
||||
{
|
||||
[DepthChartSeries.Bids]: stringGetter({ key: STRING_KEYS.BIDS }),
|
||||
[DepthChartSeries.Asks]: stringGetter({ key: STRING_KEYS.ASKS }),
|
||||
[DepthChartSeries.MidMarket]: stringGetter({ key: STRING_KEYS.MID_MARKET }),
|
||||
}[nearestDatum.key]}
|
||||
</h4>
|
||||
|
||||
<Details
|
||||
layout="column"
|
||||
items={
|
||||
isEditingOrder
|
||||
? [
|
||||
{
|
||||
key: 'side',
|
||||
label: stringGetter({ key: STRING_KEYS.SIDE }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Text}
|
||||
value={
|
||||
{
|
||||
[OrderSide.BUY]: stringGetter({
|
||||
key: STRING_KEYS.BUY,
|
||||
}),
|
||||
[OrderSide.SELL]: stringGetter({
|
||||
key: STRING_KEYS.SELL,
|
||||
}),
|
||||
}[chartPointAtPointer.side]
|
||||
}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'limitPrice',
|
||||
label: stringGetter({ key: STRING_KEYS.LIMIT_PRICE }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Fiat}
|
||||
value={chartPointAtPointer.price}
|
||||
useGrouping={false}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'size',
|
||||
label: stringGetter({ key: STRING_KEYS.AMOUNT }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Asset}
|
||||
value={chartPointAtPointer.size}
|
||||
fractionDigits={stepSizeDecimals}
|
||||
tag={id}
|
||||
useGrouping={false}
|
||||
/>
|
||||
),
|
||||
},
|
||||
]
|
||||
: nearestDatum?.key === DepthChartSeries.MidMarket
|
||||
? [
|
||||
{
|
||||
key: 'midMarketPrice',
|
||||
label: stringGetter({ key: STRING_KEYS.PRICE }),
|
||||
value: (
|
||||
<Output type={OutputType.Fiat} value={midMarketPrice} useGrouping={false} />
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'spread',
|
||||
label: stringGetter({ key: STRING_KEYS.ORDERBOOK_SPREAD }),
|
||||
value: (
|
||||
<>
|
||||
<Output
|
||||
type={OutputType.Fiat}
|
||||
value={spread}
|
||||
fractionDigits={tickSizeDecimals}
|
||||
useGrouping={false}
|
||||
/>
|
||||
<Output
|
||||
type={OutputType.SmallPercent}
|
||||
value={spreadPercent}
|
||||
withParentheses
|
||||
/>
|
||||
</>
|
||||
),
|
||||
},
|
||||
]
|
||||
: [
|
||||
{
|
||||
key: 'price',
|
||||
label: stringGetter({ key: STRING_KEYS.PRICE }),
|
||||
value: (
|
||||
<>
|
||||
{nearestDatum &&
|
||||
{
|
||||
[DepthChartSeries.Bids]: '≥',
|
||||
[DepthChartSeries.Asks]: '≤',
|
||||
}[nearestDatum.key]}
|
||||
<Output
|
||||
type={OutputType.Fiat}
|
||||
value={nearestDatum?.datum.price}
|
||||
useGrouping={false}
|
||||
/>
|
||||
</>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'depth',
|
||||
label: stringGetter({ key: STRING_KEYS.TOTAL_SIZE }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Asset}
|
||||
value={nearestDatum?.datum.depth}
|
||||
fractionDigits={stepSizeDecimals}
|
||||
tag={id}
|
||||
useGrouping={false}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'cost',
|
||||
label: stringGetter({ key: STRING_KEYS.TOTAL_COST }),
|
||||
value: (
|
||||
<Output
|
||||
useGrouping
|
||||
type={OutputType.Fiat}
|
||||
value={
|
||||
nearestDatum
|
||||
? nearestDatum?.datum.price * nearestDatum?.datum.depth
|
||||
: undefined
|
||||
}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'priceImpact',
|
||||
label: stringGetter({ key: STRING_KEYS.PRICE_IMPACT }),
|
||||
value: <Output useGrouping type={OutputType.Percent} value={priceImpact} />,
|
||||
},
|
||||
]
|
||||
}
|
||||
/>
|
||||
</TooltipContent>
|
||||
);
|
||||
};
|
||||
@ -1,14 +1,20 @@
|
||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||
import styled, { AnyStyledComponent, css, keyframes } from 'styled-components';
|
||||
import { useSelector, shallowEqual } from 'react-redux';
|
||||
import { OrderSide } from '@dydxprotocol/v4-client-js';
|
||||
|
||||
import { StringGetterFunction, STRING_KEYS } from '@/constants/localization';
|
||||
import {
|
||||
DepthChartDatum,
|
||||
DepthChartPoint,
|
||||
DepthChartSeries,
|
||||
SERIES_KEY_FOR_ORDER_SIDE,
|
||||
} from '@/constants/charts';
|
||||
import { StringGetterFunction } from '@/constants/localization';
|
||||
|
||||
import { useBreakpoints } from '@/hooks';
|
||||
import { useOrderbookValuesForDepthChart } from '@/hooks/useOrderbookValues';
|
||||
|
||||
import { MustBigNumber } from '@/lib/numbers';
|
||||
|
||||
import { getCurrentMarketConfig, getCurrentMarketOrderbook } from '@/state/perpetualsSelectors';
|
||||
import { getCurrentMarketConfig } from '@/state/perpetualsSelectors';
|
||||
import { getCurrentMarketAssetData } from '@/state/assetsSelectors';
|
||||
|
||||
import { XYChartWithPointerEvents } from '@/components/visx/XYChartWithPointerEvents';
|
||||
@ -21,24 +27,25 @@ import {
|
||||
darkTheme,
|
||||
DataProvider,
|
||||
EventEmitterProvider,
|
||||
type EventHandlerParams,
|
||||
} from '@visx/xychart';
|
||||
import { LinearGradient } from '@visx/gradient';
|
||||
import { curveStepAfter } from '@visx/curve';
|
||||
import type { Point } from '@visx/point';
|
||||
import { Point } from '@visx/point';
|
||||
import Tooltip from '@/components/visx/XYChartTooltipWithBounds';
|
||||
import { TooltipContent } from '@/components/visx/TooltipContent';
|
||||
import { AxisLabelOutput } from '@/components/visx/AxisLabelOutput';
|
||||
|
||||
import { Details } from '@/components/Details';
|
||||
import { LoadingSpace } from '@/components/Loading/LoadingSpinner';
|
||||
import { Output, OutputType } from '@/components/Output';
|
||||
import { OutputType } from '@/components/Output';
|
||||
|
||||
import { OrderSide } from '@dydxprotocol/v4-client-js';
|
||||
import { MustBigNumber } from '@/lib/numbers';
|
||||
|
||||
import { DepthChartTooltipContent } from './Tooltip';
|
||||
|
||||
// @ts-ignore
|
||||
const theme = buildChartTheme({
|
||||
...darkTheme,
|
||||
colors: ['var(--color-positive)', 'var(--color-negative)', 'white'], // categorical colors, mapped to series via `dataKey`s
|
||||
colors: ['var(--color-positive)', 'var(--color-negative)', 'var(--color-layer-6)'], // categorical colors, mapped to series via `dataKey`s
|
||||
});
|
||||
|
||||
const clamp = (n: number, min: number, max: number) => Math.max(min, Math.min(max, n));
|
||||
@ -58,30 +65,6 @@ const formatNumber = (n: number, selectedLocale: string, isCompact: boolean = n
|
||||
: formattedNumber;
|
||||
};
|
||||
|
||||
enum DepthChartSeries {
|
||||
Asks = 'Asks',
|
||||
Bids = 'Bids',
|
||||
MidMarket = 'MidMarket',
|
||||
}
|
||||
|
||||
type DepthChartDatum = {
|
||||
size: number;
|
||||
price: number;
|
||||
depth: number;
|
||||
seriesKey: DepthChartSeries;
|
||||
};
|
||||
|
||||
const seriesKeyForOrderSide = {
|
||||
[OrderSide.BUY]: DepthChartSeries.Bids,
|
||||
[OrderSide.SELL]: DepthChartSeries.Asks,
|
||||
};
|
||||
|
||||
type DepthChartPoint = {
|
||||
side: OrderSide;
|
||||
price: number;
|
||||
size: number;
|
||||
};
|
||||
|
||||
export const DepthChart = ({
|
||||
onChartClick,
|
||||
stringGetter,
|
||||
@ -96,8 +79,6 @@ export const DepthChart = ({
|
||||
const { isMobile } = useBreakpoints();
|
||||
|
||||
// Chart data
|
||||
|
||||
const orderbook = useSelector(getCurrentMarketOrderbook, shallowEqual);
|
||||
const { id = '' } = useSelector(getCurrentMarketAssetData, shallowEqual) ?? {};
|
||||
const { stepSizeDecimals, tickSizeDecimals } =
|
||||
useSelector(getCurrentMarketConfig, shallowEqual) ?? {};
|
||||
@ -112,43 +93,15 @@ export const DepthChart = ({
|
||||
midMarketPrice,
|
||||
spread,
|
||||
spreadPercent,
|
||||
} = useMemo(() => {
|
||||
const bids = (orderbook?.bids?.toArray() ?? [])
|
||||
.filter(Boolean)
|
||||
.map((datum) => ({ ...datum, seriesKey: DepthChartSeries.Bids } as DepthChartDatum));
|
||||
|
||||
const asks = (orderbook?.asks?.toArray() ?? [])
|
||||
.filter(Boolean)
|
||||
.map((datum) => ({ ...datum, seriesKey: DepthChartSeries.Asks } as DepthChartDatum));
|
||||
|
||||
const lowestBid = bids[bids.length - 1];
|
||||
const highestBid = bids[0];
|
||||
const lowestAsk = asks[0];
|
||||
const highestAsk = asks[asks.length - 1];
|
||||
|
||||
const midMarketPrice = orderbook?.midPrice;
|
||||
const spread = MustBigNumber(lowestAsk?.price ?? 0).minus(highestBid?.price ?? 0);
|
||||
const spreadPercent = orderbook?.spreadPercent;
|
||||
|
||||
return {
|
||||
bids,
|
||||
asks,
|
||||
lowestBid,
|
||||
highestBid,
|
||||
lowestAsk,
|
||||
highestAsk,
|
||||
midMarketPrice,
|
||||
spread,
|
||||
spreadPercent,
|
||||
};
|
||||
}, [orderbook]);
|
||||
orderbook,
|
||||
} = useOrderbookValuesForDepthChart();
|
||||
|
||||
// Chart state
|
||||
|
||||
const [isPointerPressed, setIsPointerPressed] = useState(false);
|
||||
const [chartPointAtPointer, setChartPointAtPointer] = useState<DepthChartPoint>();
|
||||
|
||||
const isEditingOrder = isPointerPressed && chartPointAtPointer;
|
||||
const isEditingOrder = Boolean(isPointerPressed && chartPointAtPointer);
|
||||
|
||||
const [zoomDomain, setZoomDomain] = useState<undefined | number>();
|
||||
|
||||
@ -193,12 +146,24 @@ export const DepthChart = ({
|
||||
}, [orderbook, zoomDomain]);
|
||||
|
||||
const getChartPoint = useCallback(
|
||||
({ x: price, y: size }: Point) =>
|
||||
({
|
||||
side: price < midMarketPrice! ? OrderSide.BUY : OrderSide.SELL,
|
||||
(point: Point | EventHandlerParams<object>) => {
|
||||
let price, size;
|
||||
if (point instanceof Point) {
|
||||
const { x, y } = point as Point;
|
||||
price = x;
|
||||
size = y;
|
||||
} else {
|
||||
const { svgPoint: { x, y } = {} } = point as EventHandlerParams<object>;
|
||||
price = x;
|
||||
size = y;
|
||||
}
|
||||
|
||||
return {
|
||||
side: MustBigNumber(price).lt(midMarketPrice!) ? OrderSide.BUY : OrderSide.SELL,
|
||||
price,
|
||||
size,
|
||||
} as DepthChartPoint),
|
||||
} as DepthChartPoint;
|
||||
},
|
||||
[midMarketPrice]
|
||||
);
|
||||
|
||||
@ -256,7 +221,7 @@ export const DepthChart = ({
|
||||
}}
|
||||
onPointerUp={(point) => point && onChartClick?.(getChartPoint(point))}
|
||||
onPointerMove={(point) => point && setChartPointAtPointer(getChartPoint(point))}
|
||||
onPointerPressedChange={setIsPointerPressed}
|
||||
onPointerPressedChange={(isPointerPressed) => setIsPointerPressed(isPointerPressed)}
|
||||
>
|
||||
<Axis
|
||||
orientation="bottom"
|
||||
@ -363,7 +328,7 @@ export const DepthChart = ({
|
||||
<Styled.XAxisLabelOutput
|
||||
type={OutputType.Fiat}
|
||||
value={
|
||||
isEditingOrder
|
||||
isEditingOrder && chartPointAtPointer
|
||||
? chartPointAtPointer.price
|
||||
: tooltipData!.nearestDatum?.datum.price
|
||||
}
|
||||
@ -374,8 +339,8 @@ export const DepthChart = ({
|
||||
[DepthChartSeries.Bids]: 'var(--color-positive)',
|
||||
[DepthChartSeries.MidMarket]: 'var(--color-layer-6)',
|
||||
}[
|
||||
isEditingOrder
|
||||
? seriesKeyForOrderSide[chartPointAtPointer.side]
|
||||
isEditingOrder && chartPointAtPointer
|
||||
? SERIES_KEY_FOR_ORDER_SIDE[chartPointAtPointer.side]
|
||||
: (tooltipData!.nearestDatum?.key as DepthChartSeries)
|
||||
]
|
||||
}
|
||||
@ -389,7 +354,7 @@ export const DepthChart = ({
|
||||
<Styled.YAxisLabelOutput
|
||||
type={OutputType.Asset}
|
||||
value={
|
||||
isEditingOrder
|
||||
isEditingOrder && chartPointAtPointer
|
||||
? chartPointAtPointer.size
|
||||
: tooltipData!.nearestDatum?.datum.depth
|
||||
}
|
||||
@ -400,8 +365,8 @@ export const DepthChart = ({
|
||||
[DepthChartSeries.Bids]: 'var(--color-positive)',
|
||||
[DepthChartSeries.MidMarket]: 'var(--color-layer-6)',
|
||||
}[
|
||||
isEditingOrder
|
||||
? seriesKeyForOrderSide[chartPointAtPointer.side]
|
||||
isEditingOrder && chartPointAtPointer
|
||||
? SERIES_KEY_FOR_ORDER_SIDE[chartPointAtPointer.side]
|
||||
: (tooltipData!.nearestDatum?.key as DepthChartSeries)
|
||||
]
|
||||
}
|
||||
@ -410,181 +375,22 @@ export const DepthChart = ({
|
||||
}
|
||||
snapTooltipToDatumX={!isEditingOrder}
|
||||
snapTooltipToDatumY={isEditingOrder ? false : isMobile}
|
||||
renderTooltip={({ tooltipData, colorScale }) => {
|
||||
const { nearestDatum } = tooltipData || {};
|
||||
|
||||
if (!isEditingOrder && !nearestDatum?.datum) return null;
|
||||
|
||||
return (
|
||||
<TooltipContent
|
||||
accentColor={colorScale?.(
|
||||
isEditingOrder
|
||||
? seriesKeyForOrderSide[chartPointAtPointer.side]
|
||||
: nearestDatum.key
|
||||
)}
|
||||
>
|
||||
<h4>
|
||||
{isEditingOrder
|
||||
? 'Release mouse to edit order'
|
||||
: {
|
||||
[DepthChartSeries.Bids]: 'Bids',
|
||||
[DepthChartSeries.Asks]: 'Asks',
|
||||
[DepthChartSeries.MidMarket]: 'Mid-Market',
|
||||
}[nearestDatum.key]}
|
||||
</h4>
|
||||
|
||||
<Details
|
||||
layout="column"
|
||||
items={
|
||||
isEditingOrder
|
||||
? [
|
||||
{
|
||||
key: 'side',
|
||||
label: stringGetter({ key: STRING_KEYS.SIDE }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Text}
|
||||
value={
|
||||
{
|
||||
[OrderSide.BUY]: stringGetter({
|
||||
key: STRING_KEYS.BUY,
|
||||
}),
|
||||
[OrderSide.SELL]: stringGetter({
|
||||
key: STRING_KEYS.SELL,
|
||||
}),
|
||||
}[chartPointAtPointer.side]
|
||||
}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'limitPrice',
|
||||
label: stringGetter({ key: STRING_KEYS.LIMIT_PRICE }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Fiat}
|
||||
value={chartPointAtPointer.price}
|
||||
useGrouping={false}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'size',
|
||||
label: stringGetter({ key: STRING_KEYS.AMOUNT }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Asset}
|
||||
value={chartPointAtPointer.size}
|
||||
fractionDigits={stepSizeDecimals}
|
||||
tag={id}
|
||||
useGrouping={false}
|
||||
/>
|
||||
),
|
||||
},
|
||||
]
|
||||
: nearestDatum?.key === DepthChartSeries.MidMarket
|
||||
? [
|
||||
{
|
||||
key: 'midMarketPrice',
|
||||
// label: stringGetter({ key: STRING_KEYS.ORDERBOOK_MID_MARKET_PRICE }),
|
||||
label: stringGetter({ key: STRING_KEYS.PRICE }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Fiat}
|
||||
value={midMarketPrice}
|
||||
useGrouping={false}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'spread',
|
||||
label: stringGetter({ key: STRING_KEYS.ORDERBOOK_SPREAD }),
|
||||
value: (
|
||||
<>
|
||||
<Output
|
||||
type={OutputType.Fiat}
|
||||
value={spread}
|
||||
fractionDigits={tickSizeDecimals}
|
||||
useGrouping={false}
|
||||
/>
|
||||
<Output
|
||||
type={OutputType.SmallPercent}
|
||||
value={spreadPercent}
|
||||
withParentheses
|
||||
/>
|
||||
</>
|
||||
),
|
||||
},
|
||||
]
|
||||
: [
|
||||
{
|
||||
key: 'price',
|
||||
label: stringGetter({ key: STRING_KEYS.PRICE }),
|
||||
value: (
|
||||
<>
|
||||
{nearestDatum &&
|
||||
{
|
||||
[DepthChartSeries.Bids]: '≥',
|
||||
[DepthChartSeries.Asks]: '≤',
|
||||
}[nearestDatum.key]}
|
||||
<Output
|
||||
type={OutputType.Fiat}
|
||||
value={nearestDatum?.datum.price}
|
||||
useGrouping={false}
|
||||
/>
|
||||
</>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'depth',
|
||||
label: stringGetter({ key: STRING_KEYS.TOTAL_SIZE }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Asset}
|
||||
value={nearestDatum?.datum.depth}
|
||||
fractionDigits={stepSizeDecimals}
|
||||
tag={id}
|
||||
useGrouping={false}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'cost',
|
||||
label: stringGetter({ key: STRING_KEYS.TOTAL_COST }),
|
||||
value: (
|
||||
<Output
|
||||
useGrouping
|
||||
type={OutputType.Fiat}
|
||||
value={nearestDatum?.datum.price * nearestDatum?.datum.depth}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'priceImpact',
|
||||
label: stringGetter({ key: STRING_KEYS.PRICE_IMPACT }),
|
||||
value: (
|
||||
<Output
|
||||
useGrouping
|
||||
type={OutputType.Percent}
|
||||
value={{
|
||||
[DepthChartSeries.Asks]: () =>
|
||||
MustBigNumber(nearestDatum.datum.price)
|
||||
.minus(lowestAsk.price)
|
||||
.div(nearestDatum.datum.price),
|
||||
[DepthChartSeries.Bids]: () =>
|
||||
MustBigNumber(highestBid.price)
|
||||
.minus(nearestDatum.datum.price)
|
||||
.div(highestBid.price),
|
||||
}[nearestDatum.key]()}
|
||||
/>
|
||||
),
|
||||
},
|
||||
]
|
||||
}
|
||||
/>
|
||||
</TooltipContent>
|
||||
);
|
||||
}}
|
||||
renderTooltip={({ tooltipData, colorScale }) =>
|
||||
chartPointAtPointer && (
|
||||
<DepthChartTooltipContent
|
||||
{...{
|
||||
tooltipData,
|
||||
colorScale,
|
||||
isEditingOrder,
|
||||
chartPointAtPointer,
|
||||
stringGetter,
|
||||
selectedLocale,
|
||||
stepSizeDecimals,
|
||||
tickSizeDecimals,
|
||||
}}
|
||||
/>
|
||||
)
|
||||
}
|
||||
/>
|
||||
</XYChartWithPointerEvents>
|
||||
</EventEmitterProvider>
|
||||
107
src/views/charts/FundingChart/Tooltip.tsx
Normal file
107
src/views/charts/FundingChart/Tooltip.tsx
Normal file
@ -0,0 +1,107 @@
|
||||
import type { RenderTooltipParams } from '@visx/xychart/lib/components/Tooltip';
|
||||
import { TooltipContent } from '@/components/visx/TooltipContent';
|
||||
|
||||
import { FundingRateResolution, type FundingChartDatum } from '@/constants/charts';
|
||||
import { STRING_KEYS } from '@/constants/localization';
|
||||
import { FundingDirection } from '@/constants/markets';
|
||||
import { useStringGetter } from '@/hooks';
|
||||
|
||||
import { Details, DetailsItem } from '@/components/Details';
|
||||
import { Output, OutputType, ShowSign } from '@/components/Output';
|
||||
|
||||
type FundingChartTooltipProps = {
|
||||
fundingRateView: FundingRateResolution;
|
||||
latestDatum: FundingChartDatum;
|
||||
} & Pick<RenderTooltipParams<FundingChartDatum>, 'tooltipData'>;
|
||||
|
||||
export const FundingChartTooltipContent = ({
|
||||
fundingRateView,
|
||||
latestDatum,
|
||||
tooltipData,
|
||||
}: FundingChartTooltipProps) => {
|
||||
const { nearestDatum } = tooltipData || {};
|
||||
const stringGetter = useStringGetter();
|
||||
|
||||
const tooltipDatum = nearestDatum?.datum ?? latestDatum;
|
||||
const isShowingCurrentFundingRate = tooltipDatum === latestDatum;
|
||||
|
||||
return (
|
||||
<TooltipContent
|
||||
accentColor={
|
||||
{
|
||||
[FundingDirection.ToLong]: 'var(--color-negative)',
|
||||
[FundingDirection.ToShort]: 'var(--color-positive)',
|
||||
[FundingDirection.None]: 'var(--color-layer-6)',
|
||||
}[tooltipDatum.direction]
|
||||
}
|
||||
>
|
||||
<h4>
|
||||
{isShowingCurrentFundingRate
|
||||
? stringGetter({ key: STRING_KEYS.CURRENT_FUNDING_RATE })
|
||||
: stringGetter({ key: STRING_KEYS.HISTORICAL_FUNDING_RATE })}
|
||||
</h4>
|
||||
|
||||
<Details
|
||||
layout="column"
|
||||
items={
|
||||
[
|
||||
{
|
||||
key: 'direction',
|
||||
label: stringGetter({ key: STRING_KEYS.DIRECTION }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Text}
|
||||
value={
|
||||
{
|
||||
[FundingDirection.ToLong]: `${stringGetter({
|
||||
key: STRING_KEYS.SHORT_POSITION_SHORT,
|
||||
})} → ${stringGetter({
|
||||
key: STRING_KEYS.LONG_POSITION_SHORT,
|
||||
})}`,
|
||||
[FundingDirection.ToShort]: `${stringGetter({
|
||||
key: STRING_KEYS.LONG_POSITION_SHORT,
|
||||
})} → ${stringGetter({
|
||||
key: STRING_KEYS.SHORT_POSITION_SHORT,
|
||||
})}`,
|
||||
[FundingDirection.None]: undefined,
|
||||
}[tooltipDatum.direction]
|
||||
}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'fundingRate',
|
||||
label: stringGetter({
|
||||
key: {
|
||||
[FundingRateResolution.OneHour]: STRING_KEYS.RATE_1H,
|
||||
[FundingRateResolution.EightHour]: STRING_KEYS.RATE_8H,
|
||||
[FundingRateResolution.Annualized]: STRING_KEYS.ANNUALIZED,
|
||||
}[fundingRateView],
|
||||
}),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.SmallPercent}
|
||||
value={
|
||||
{
|
||||
[FundingRateResolution.OneHour]: tooltipDatum.fundingRate,
|
||||
[FundingRateResolution.EightHour]: tooltipDatum.fundingRate * 8,
|
||||
[FundingRateResolution.Annualized]: tooltipDatum.fundingRate * (24 * 365),
|
||||
}[fundingRateView]
|
||||
}
|
||||
showSign={ShowSign.Both}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'time',
|
||||
label: isShowingCurrentFundingRate
|
||||
? 'Time Remaining'
|
||||
: stringGetter({ key: STRING_KEYS.TIME }),
|
||||
value: <Output type={OutputType.DateTime} value={tooltipDatum.time} />,
|
||||
},
|
||||
].filter(Boolean) as Array<DetailsItem>
|
||||
}
|
||||
/>
|
||||
</TooltipContent>
|
||||
);
|
||||
};
|
||||
@ -3,39 +3,27 @@ import { shallowEqual, useSelector } from 'react-redux';
|
||||
import styled, { type AnyStyledComponent, css } from 'styled-components';
|
||||
import { curveMonotoneX, curveStepAfter } from '@visx/curve';
|
||||
|
||||
import { useBreakpoints, useStringGetter } from '@/hooks';
|
||||
import { ButtonSize } from '@/constants/buttons';
|
||||
import { FundingRateResolution, type FundingChartDatum } from '@/constants/charts';
|
||||
import { STRING_KEYS } from '@/constants/localization';
|
||||
import { SMALL_PERCENT_DECIMALS, TINY_PERCENT_DECIMALS } from '@/constants/numbers';
|
||||
import { FundingDirection } from '@/constants/markets';
|
||||
import { SMALL_PERCENT_DECIMALS, TINY_PERCENT_DECIMALS } from '@/constants/numbers';
|
||||
import { useBreakpoints, useStringGetter } from '@/hooks';
|
||||
|
||||
import { breakpoints } from '@/styles';
|
||||
|
||||
import { Details, DetailsItem } from '@/components/Details';
|
||||
import { Output, OutputType, ShowSign } from '@/components/Output';
|
||||
import { Output, OutputType } from '@/components/Output';
|
||||
import { LoadingSpace } from '@/components/Loading/LoadingSpinner';
|
||||
import { ToggleGroup } from '@/components/ToggleGroup';
|
||||
|
||||
import { TimeSeriesChart } from '@/components/visx/TimeSeriesChart';
|
||||
import { AxisLabelOutput } from '@/components/visx/AxisLabelOutput';
|
||||
import { TooltipContent } from '@/components/visx/TooltipContent';
|
||||
import type { TooltipContextType } from '@visx/xychart';
|
||||
|
||||
import { calculateFundingRateHistory } from '@/state/perpetualsCalculators';
|
||||
|
||||
import { MustBigNumber } from '@/lib/numbers';
|
||||
|
||||
enum FundingRateResolution {
|
||||
OneHour = 'OneHour',
|
||||
EightHour = 'EightHour',
|
||||
Annualized = 'Annualized',
|
||||
}
|
||||
|
||||
type FundingChartDatum = {
|
||||
time: number;
|
||||
fundingRate: number;
|
||||
direction: FundingDirection;
|
||||
};
|
||||
import { FundingChartTooltipContent } from './Tooltip';
|
||||
|
||||
const FUNDING_RATE_TIME_RESOLUTION = 60 * 60 * 1000; // 1 hour
|
||||
|
||||
@ -116,114 +104,20 @@ export const FundingChart = ({ selectedLocale }: ElementProps) => {
|
||||
{
|
||||
[FundingDirection.ToLong]: 'var(--color-negative)',
|
||||
[FundingDirection.ToShort]: 'var(--color-positive)',
|
||||
[FundingDirection.None]: 'var(--color-layer-6)',
|
||||
}[tooltipDatum.direction]
|
||||
}
|
||||
/>
|
||||
);
|
||||
}}
|
||||
renderTooltip={({ tooltipData }) => {
|
||||
const { nearestDatum } = tooltipData || {};
|
||||
|
||||
const tooltipDatum = nearestDatum?.datum ?? latestDatum;
|
||||
const isShowingCurrentFundingRate = tooltipDatum === latestDatum;
|
||||
|
||||
return (
|
||||
<TooltipContent
|
||||
accentColor={
|
||||
{
|
||||
[FundingDirection.ToLong]: 'var(--color-negative)',
|
||||
[FundingDirection.ToShort]: 'var(--color-positive)',
|
||||
}[tooltipDatum.direction]
|
||||
}
|
||||
>
|
||||
<h4>
|
||||
{isShowingCurrentFundingRate
|
||||
? stringGetter({ key: STRING_KEYS.CURRENT_FUNDING_RATE })
|
||||
: stringGetter({ key: STRING_KEYS.HISTORICAL_FUNDING_RATE })}
|
||||
</h4>
|
||||
|
||||
<Details
|
||||
layout="column"
|
||||
items={
|
||||
[
|
||||
{
|
||||
key: 'direction',
|
||||
label: stringGetter({ key: STRING_KEYS.DIRECTION }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.Text}
|
||||
value={
|
||||
{
|
||||
[FundingDirection.ToLong]: `${stringGetter({
|
||||
key: STRING_KEYS.SHORT_POSITION_SHORT,
|
||||
})} → ${stringGetter({
|
||||
key: STRING_KEYS.LONG_POSITION_SHORT,
|
||||
})}`,
|
||||
[FundingDirection.ToShort]: `${stringGetter({
|
||||
key: STRING_KEYS.LONG_POSITION_SHORT,
|
||||
})} → ${stringGetter({
|
||||
key: STRING_KEYS.SHORT_POSITION_SHORT,
|
||||
})}`,
|
||||
}[tooltipDatum.direction]
|
||||
}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'fundingRate1h',
|
||||
label: stringGetter({ key: STRING_KEYS.RATE_1H }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.SmallPercent}
|
||||
value={tooltipDatum.fundingRate}
|
||||
showSign={ShowSign.Both}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'fundingRate8h',
|
||||
label: stringGetter({ key: STRING_KEYS.RATE_8H }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.SmallPercent}
|
||||
value={tooltipDatum.fundingRate * 8}
|
||||
// value={
|
||||
// Math.sign(tooltipDatum.fundingRate) *
|
||||
// ((Math.abs(tooltipDatum.fundingRate) + 1) ** 8 - 1)
|
||||
// }
|
||||
showSign={ShowSign.Both}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'fundingRateAnnualized',
|
||||
label: stringGetter({ key: STRING_KEYS.ANNUALIZED }),
|
||||
value: (
|
||||
<Output
|
||||
type={OutputType.SmallPercent}
|
||||
value={tooltipDatum.fundingRate * (24 * 365)}
|
||||
// value={
|
||||
// Math.sign(tooltipDatum.fundingRate) *
|
||||
// ((Math.abs(tooltipDatum.fundingRate) + 1) ** (24 * 365) - 1)
|
||||
// }
|
||||
showSign={ShowSign.Both}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'time',
|
||||
label: isShowingCurrentFundingRate
|
||||
? 'Time Remaining'
|
||||
: stringGetter({ key: STRING_KEYS.TIME }),
|
||||
value: <Output type={OutputType.DateTime} value={tooltipDatum.time} />,
|
||||
},
|
||||
].filter(Boolean) as Array<DetailsItem>
|
||||
}
|
||||
/>
|
||||
</TooltipContent>
|
||||
);
|
||||
}}
|
||||
onTooltipContext={setTooltipContext}
|
||||
renderTooltip={({ tooltipData }) => (
|
||||
<FundingChartTooltipContent
|
||||
fundingRateView={fundingRateView}
|
||||
tooltipData={tooltipData}
|
||||
latestDatum={latestDatum}
|
||||
/>
|
||||
)}
|
||||
onTooltipContext={(tooltipContext) => setTooltipContext(tooltipContext)}
|
||||
minZoomDomain={FUNDING_RATE_TIME_RESOLUTION * 4}
|
||||
numGridLines={1}
|
||||
slotEmpty={<LoadingSpace id="funding-chart-loading" />}
|
||||
@ -290,6 +184,7 @@ Styled.FundingRateToggle = styled.div`
|
||||
Styled.CurrentFundingRate = styled.div<{ isShowing?: boolean }>`
|
||||
place-self: start center;
|
||||
padding: clamp(1.5rem, 9rem - 15%, 4rem);
|
||||
pointer-events: none;
|
||||
|
||||
font: var(--font-large-book);
|
||||
|
||||
Loading…
Reference in New Issue
Block a user