import { gql, useQuery } from '@apollo/client'; import { useAssetDetailsDialogStore } from '@vegaprotocol/assets'; import { addDecimalsFormatNumber, t, titlefy, } from '@vegaprotocol/react-helpers'; import { Interval } from '@vegaprotocol/types'; import { AsyncRenderer, Splash } from '@vegaprotocol/ui-toolkit'; import debounce from 'lodash/debounce'; import { useRouter } from 'next/router'; import React, { useEffect, useMemo, useState } from 'react'; import { useGlobalStore } from '../../stores'; import { TradeGrid, TradePanels } from './trade-grid'; import type { Market, MarketVariables } from './__generated__/Market'; import { ColumnKind, SelectMarketDialog } from '../../components/select-market'; // Top level page query const MARKET_QUERY = gql` query Market($marketId: ID!, $interval: Interval!, $since: String!) { market(id: $marketId) { id tradingMode state decimalPlaces positionDecimalPlaces data { market { id } auctionStart auctionEnd markPrice indicativeVolume indicativePrice suppliedStake targetStake bestBidVolume bestOfferVolume bestStaticBidVolume bestStaticOfferVolume trigger } tradableInstrument { instrument { id name code metadata { tags } product { ... on Future { oracleSpecForTradingTermination { id } quoteName settlementAsset { id symbol name decimals } } } } } marketTimestamps { open close } depth { lastTrade { price } } candlesConnection(interval: $interval, since: $since) { edges { node { open close volume } } } } } `; const MarketPage = ({ id }: { id?: string }) => { const { query, push } = useRouter(); const { w } = useWindowSize(); const { landingDialog, riskNoticeDialog, update } = useGlobalStore( (store) => ({ landingDialog: store.landingDialog, riskNoticeDialog: store.riskNoticeDialog, update: store.update, }) ); const { update: updateStore } = useGlobalStore((store) => ({ update: store.update, })); const { open: openAssetDetailsDialog } = useAssetDetailsDialogStore(); // Default to first marketId query item if found const marketId = id || (Array.isArray(query.marketId) ? query.marketId[0] : query.marketId); const onSelect = (id: string) => { if (id && id !== marketId) { updateStore({ marketId: id }); push(`/markets/${id}`); } }; // Cache timestamp for yesterday to prevent full unmount of market page when // a rerender occurs const yTimestamp = useMemo(() => { const yesterday = Math.round(new Date().getTime() / 1000) - 24 * 3600; return new Date(yesterday * 1000).toISOString(); }, []); const variables = useMemo( () => ({ marketId: marketId || '', interval: Interval.INTERVAL_I1H, since: yTimestamp, }), [marketId, yTimestamp] ); const { data, error, loading } = useQuery( MARKET_QUERY, { variables, fetchPolicy: 'network-only', errorPolicy: 'ignore', } ); useEffect(() => { const marketName = data?.market?.tradableInstrument.instrument.name; const marketPrice = data?.market && data?.market?.data ? addDecimalsFormatNumber( data.market.data.markPrice, data.market.decimalPlaces ) : null; if (marketName) { const pageTitle = titlefy([marketName, marketPrice]); update({ pageTitle }); } }, [data, update]); if (!marketId) { return (

{t('Not found')}

); } return ( loading={loading} error={error} data={data} render={({ market }) => { if (!market) { return {t('Market not found')}; } return ( <> {w > 960 ? ( ) : ( )} update({ landingDialog: isOpen }) } onSelect={onSelect} onCellClick={(e, kind, value) => { if (value && kind === ColumnKind.Asset) { openAssetDetailsDialog(value, e.target as HTMLElement); } }} /> ); }} /> ); }; MarketPage.getInitialProps = () => ({ page: 'market', }); export default MarketPage; const useWindowSize = () => { const [windowSize, setWindowSize] = useState(() => { if (typeof window !== 'undefined') { return { w: window.innerWidth, h: window.innerHeight, }; } // Something sensible for server rendered page return { w: 1200, h: 900, }; }); useEffect(() => { const handleResize = debounce(({ target }) => { setWindowSize({ w: target.innerWidth, h: target.innerHeight, }); }, 300); window.addEventListener('resize', handleResize); return () => { window.removeEventListener('resize', handleResize); }; }, []); return windowSize; };