2023-12-06 13:31:40 +00:00
|
|
|
import { type PinnedAsset } from '@vegaprotocol/accounts';
|
|
|
|
import { type Market } from '@vegaprotocol/markets';
|
2023-05-18 11:22:54 +00:00
|
|
|
import { OracleBanner } from '@vegaprotocol/markets';
|
2023-08-10 15:02:46 +00:00
|
|
|
import { useState } from 'react';
|
2023-05-16 16:57:36 +00:00
|
|
|
import AutoSizer from 'react-virtualized-auto-sizer';
|
|
|
|
import classNames from 'classnames';
|
2023-12-06 13:31:40 +00:00
|
|
|
import { FLAGS } from '@vegaprotocol/environment';
|
|
|
|
import { Splash } from '@vegaprotocol/ui-toolkit';
|
|
|
|
import { useT } from '../../lib/use-t';
|
2023-07-27 14:54:00 +00:00
|
|
|
import {
|
|
|
|
MarketSuccessorBanner,
|
|
|
|
MarketSuccessorProposalBanner,
|
2023-10-10 15:50:49 +00:00
|
|
|
MarketTerminationBanner,
|
2023-07-27 14:54:00 +00:00
|
|
|
} from '../../components/market-banner';
|
2023-12-06 13:31:40 +00:00
|
|
|
import { ErrorBoundary } from '../../components/error-boundary';
|
|
|
|
import { type TradingView } from './trade-views';
|
|
|
|
import { TradingViews } from './trade-views';
|
2023-05-16 16:57:36 +00:00
|
|
|
|
|
|
|
interface TradePanelsProps {
|
|
|
|
market: Market | null;
|
|
|
|
pinnedAsset?: PinnedAsset;
|
|
|
|
}
|
|
|
|
|
2023-08-10 15:02:46 +00:00
|
|
|
export const TradePanels = ({ market, pinnedAsset }: TradePanelsProps) => {
|
2023-05-16 16:57:36 +00:00
|
|
|
const [view, setView] = useState<TradingView>('candles');
|
2023-08-02 14:29:41 +00:00
|
|
|
|
2023-05-16 16:57:36 +00:00
|
|
|
const renderView = () => {
|
2023-08-10 15:02:46 +00:00
|
|
|
const Component = TradingViews[view].component;
|
2023-05-16 16:57:36 +00:00
|
|
|
|
|
|
|
if (!Component) {
|
|
|
|
throw new Error(`No component for view: ${view}`);
|
|
|
|
}
|
|
|
|
|
2023-11-16 03:10:39 +00:00
|
|
|
if (!market) return <NoMarketSplash />;
|
2023-05-16 16:57:36 +00:00
|
|
|
|
2023-11-16 03:10:39 +00:00
|
|
|
// Watch out here, we don't know what component is being rendered
|
|
|
|
// so watch out for clashes in props
|
2023-12-06 13:31:40 +00:00
|
|
|
return (
|
|
|
|
<ErrorBoundary feature={view}>
|
|
|
|
<Component marketId={market?.id} pinnedAsset={pinnedAsset} />;
|
|
|
|
</ErrorBoundary>
|
|
|
|
);
|
2023-05-16 16:57:36 +00:00
|
|
|
};
|
|
|
|
|
2023-08-02 14:29:41 +00:00
|
|
|
const renderMenu = () => {
|
|
|
|
const viewCfg = TradingViews[view];
|
|
|
|
|
|
|
|
if ('menu' in viewCfg) {
|
|
|
|
const Menu = viewCfg.menu;
|
2023-08-10 15:02:46 +00:00
|
|
|
|
2023-08-02 14:29:41 +00:00
|
|
|
return (
|
|
|
|
<div className="flex gap-1 p-1 bg-vega-clight-800 dark:bg-vega-cdark-800 border-b border-default">
|
2023-10-23 06:34:50 +00:00
|
|
|
<Menu />
|
2023-08-02 14:29:41 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
};
|
|
|
|
|
2023-05-16 16:57:36 +00:00
|
|
|
return (
|
2023-08-02 14:29:41 +00:00
|
|
|
<div className="h-full grid grid-rows-[min-content_min-content_1fr_min-content]">
|
2023-05-16 16:57:36 +00:00
|
|
|
<div>
|
2023-07-27 14:54:00 +00:00
|
|
|
{FLAGS.SUCCESSOR_MARKETS && (
|
|
|
|
<>
|
|
|
|
<MarketSuccessorBanner market={market} />
|
|
|
|
<MarketSuccessorProposalBanner marketId={market?.id} />
|
|
|
|
</>
|
|
|
|
)}
|
2023-10-10 15:50:49 +00:00
|
|
|
<MarketTerminationBanner market={market} />
|
2023-05-16 16:57:36 +00:00
|
|
|
<OracleBanner marketId={market?.id || ''} />
|
|
|
|
</div>
|
2023-08-02 14:29:41 +00:00
|
|
|
<div>{renderMenu()}</div>
|
2023-05-16 16:57:36 +00:00
|
|
|
<div className="h-full">
|
|
|
|
<AutoSizer>
|
|
|
|
{({ width, height }) => (
|
|
|
|
<div style={{ width, height }} className="overflow-auto">
|
|
|
|
{renderView()}
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</AutoSizer>
|
|
|
|
</div>
|
|
|
|
<div className="flex flex-nowrap overflow-x-auto max-w-full border-t border-default">
|
2023-11-16 03:10:39 +00:00
|
|
|
{Object.keys(TradingViews)
|
|
|
|
// filter to control available views for the current market
|
|
|
|
// eg only perps should get the funding views
|
|
|
|
.filter((_key) => {
|
|
|
|
const key = _key as TradingView;
|
|
|
|
const perpOnlyViews = ['funding', 'fundingPayments'];
|
|
|
|
|
|
|
|
if (
|
|
|
|
market?.tradableInstrument.instrument.product.__typename ===
|
|
|
|
'Perpetual'
|
|
|
|
) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (perpOnlyViews.includes(key)) {
|
|
|
|
return false;
|
2023-07-31 16:08:55 +00:00
|
|
|
}
|
2023-11-16 03:10:39 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
})
|
|
|
|
.map((_key) => {
|
|
|
|
const key = _key as TradingView;
|
|
|
|
const isActive = view === key;
|
|
|
|
return (
|
|
|
|
<ViewButton
|
|
|
|
key={key}
|
|
|
|
view={key}
|
|
|
|
isActive={isActive}
|
|
|
|
onClick={() => setView(key)}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
})}
|
2023-05-16 16:57:36 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
2023-11-16 03:10:39 +00:00
|
|
|
|
|
|
|
export const NoMarketSplash = () => {
|
|
|
|
const t = useT();
|
|
|
|
return <Splash>{t('No market')}</Splash>;
|
|
|
|
};
|
|
|
|
|
|
|
|
const ViewButton = ({
|
|
|
|
view,
|
|
|
|
isActive,
|
|
|
|
onClick,
|
|
|
|
}: {
|
|
|
|
view: TradingView;
|
|
|
|
isActive: boolean;
|
|
|
|
onClick: () => void;
|
|
|
|
}) => {
|
|
|
|
const label = useViewLabel(view);
|
|
|
|
const className = classNames('py-2 px-4 min-w-[100px] capitalize text-sm', {
|
|
|
|
'bg-vega-clight-500 dark:bg-vega-cdark-500': isActive,
|
|
|
|
});
|
|
|
|
|
|
|
|
return (
|
|
|
|
<button data-testid={view} onClick={onClick} className={className}>
|
|
|
|
{label}
|
|
|
|
</button>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const useViewLabel = (view: TradingView) => {
|
|
|
|
const t = useT();
|
|
|
|
|
|
|
|
const labels = {
|
|
|
|
candles: t('Candles'),
|
|
|
|
depth: t('Depth'),
|
|
|
|
liquidity: t('Liquidity'),
|
|
|
|
funding: t('Funding'),
|
|
|
|
fundingPayments: t('Funding Payments'),
|
|
|
|
orderbook: t('Orderbook'),
|
|
|
|
trades: t('Trades'),
|
|
|
|
positions: t('Positions'),
|
|
|
|
activeOrders: t('Active'),
|
|
|
|
closedOrders: t('Closed'),
|
|
|
|
rejectedOrders: t('Rejected'),
|
|
|
|
orders: t('All'),
|
|
|
|
stopOrders: t('Stop'),
|
|
|
|
collateral: t('Collateral'),
|
|
|
|
fills: t('Fills'),
|
|
|
|
};
|
|
|
|
|
|
|
|
return labels[view];
|
|
|
|
};
|