2023-10-23 11:42:37 +00:00
|
|
|
import type { ReactNode } from 'react';
|
2023-10-10 15:50:49 +00:00
|
|
|
import { useState } from 'react';
|
|
|
|
import { format, formatDuration, intervalToDuration } from 'date-fns';
|
2023-10-23 11:42:37 +00:00
|
|
|
import {
|
|
|
|
ExternalLink,
|
|
|
|
Intent,
|
|
|
|
NotificationBanner,
|
|
|
|
} from '@vegaprotocol/ui-toolkit';
|
|
|
|
import type { MarketViewProposalFieldsFragment } from '@vegaprotocol/proposals';
|
|
|
|
import { marketViewProposalsDataProvider } from '@vegaprotocol/proposals';
|
2023-10-10 15:50:49 +00:00
|
|
|
import * as Types from '@vegaprotocol/types';
|
|
|
|
import type { Market } from '@vegaprotocol/markets';
|
|
|
|
import { getQuoteName } from '@vegaprotocol/markets';
|
|
|
|
import { addDecimalsFormatNumber } from '@vegaprotocol/utils';
|
2023-10-23 11:42:37 +00:00
|
|
|
import sortBy from 'lodash/sortBy';
|
|
|
|
import {
|
|
|
|
DApp,
|
|
|
|
TOKEN_PROPOSAL,
|
|
|
|
TOKEN_PROPOSALS,
|
|
|
|
useLinks,
|
|
|
|
} from '@vegaprotocol/environment';
|
|
|
|
import { useDataProvider } from '@vegaprotocol/data-provider';
|
2023-11-16 03:10:39 +00:00
|
|
|
import { useT } from '../../lib/use-t';
|
2023-10-23 11:42:37 +00:00
|
|
|
|
|
|
|
const filterProposals = (
|
|
|
|
data: MarketViewProposalFieldsFragment[] | null,
|
|
|
|
marketId: string,
|
|
|
|
now: number
|
|
|
|
) =>
|
|
|
|
sortBy(
|
|
|
|
(data || []).filter(
|
|
|
|
(item) =>
|
|
|
|
item.terms.change.__typename === 'UpdateMarketState' &&
|
|
|
|
item.terms.change.market.id === marketId &&
|
|
|
|
item.terms.change.updateType ===
|
|
|
|
Types.MarketUpdateType.MARKET_STATE_UPDATE_TYPE_TERMINATE &&
|
|
|
|
item.terms.enactmentDatetime &&
|
|
|
|
new Date(item.terms.enactmentDatetime).getTime() > now
|
|
|
|
),
|
|
|
|
(item) => item.terms.enactmentDatetime
|
|
|
|
);
|
|
|
|
|
|
|
|
const getMessageVariables = (proposal: MarketViewProposalFieldsFragment) => {
|
|
|
|
const enactmentDatetime = new Date(proposal.terms.enactmentDatetime);
|
|
|
|
const date = format(enactmentDatetime, 'dd MMMM');
|
|
|
|
const duration = formatDuration(
|
|
|
|
intervalToDuration({
|
|
|
|
start: new Date(),
|
|
|
|
end: enactmentDatetime,
|
|
|
|
}),
|
|
|
|
{
|
|
|
|
format: ['days', 'hours'],
|
|
|
|
}
|
|
|
|
);
|
|
|
|
const price =
|
|
|
|
proposal.terms.change.__typename === 'UpdateMarketState'
|
|
|
|
? proposal.terms.change.price
|
|
|
|
: '';
|
|
|
|
return {
|
|
|
|
date,
|
|
|
|
duration,
|
|
|
|
price,
|
|
|
|
};
|
|
|
|
};
|
2023-10-10 15:50:49 +00:00
|
|
|
|
|
|
|
export const MarketTerminationBanner = ({
|
|
|
|
market,
|
|
|
|
}: {
|
|
|
|
market: Market | null;
|
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-10-10 15:50:49 +00:00
|
|
|
const [visible, setVisible] = useState(true);
|
|
|
|
const skip = !market || !visible;
|
2023-10-23 11:42:37 +00:00
|
|
|
const { data: passedProposalsData } = useDataProvider({
|
|
|
|
dataProvider: marketViewProposalsDataProvider,
|
|
|
|
skip,
|
|
|
|
variables: {
|
|
|
|
inState: Types.ProposalState.STATE_PASSED,
|
|
|
|
proposalType: Types.ProposalType.TYPE_UPDATE_MARKET_STATE,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
const { data: openProposalsData } = useDataProvider({
|
|
|
|
dataProvider: marketViewProposalsDataProvider,
|
2023-10-10 15:50:49 +00:00
|
|
|
skip,
|
2023-10-23 11:42:37 +00:00
|
|
|
variables: {
|
|
|
|
inState: Types.ProposalState.STATE_OPEN,
|
|
|
|
proposalType: Types.ProposalType.TYPE_UPDATE_MARKET_STATE,
|
|
|
|
},
|
2023-10-10 15:50:49 +00:00
|
|
|
});
|
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
const governanceLink = useLinks(DApp.Governance);
|
|
|
|
|
2023-10-10 15:50:49 +00:00
|
|
|
if (!market) return null;
|
2023-10-23 11:42:37 +00:00
|
|
|
const now = Date.now();
|
|
|
|
const passedProposals = filterProposals(passedProposalsData, market.id, now);
|
|
|
|
const openProposals = filterProposals(openProposalsData, market.id, now);
|
2023-10-10 15:50:49 +00:00
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
const name = market.tradableInstrument.instrument.code;
|
|
|
|
if (!passedProposals.length && !openProposals.length) {
|
|
|
|
return null;
|
|
|
|
}
|
2023-10-10 15:50:49 +00:00
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
const assetSymbol = getQuoteName(market);
|
|
|
|
const proposalLink =
|
|
|
|
!passedProposals.length && openProposals[0]?.id
|
|
|
|
? governanceLink(TOKEN_PROPOSAL.replace(':id', openProposals[0]?.id))
|
|
|
|
: undefined;
|
|
|
|
const proposalsLink =
|
|
|
|
openProposals.length > 1 ? governanceLink(TOKEN_PROPOSALS) : undefined;
|
|
|
|
let content: ReactNode;
|
|
|
|
if (passedProposals.length) {
|
|
|
|
const { date, duration, price } = getMessageVariables(passedProposals[0]);
|
|
|
|
content = (
|
|
|
|
<>
|
2023-10-10 15:50:49 +00:00
|
|
|
<div className="uppercase mb-1">
|
2023-11-16 03:10:39 +00:00
|
|
|
{t('Trading on Market {{name}} will stop on {{date}}', {
|
|
|
|
name,
|
|
|
|
date,
|
|
|
|
})}
|
2023-10-10 15:50:49 +00:00
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
{t(
|
2023-11-16 03:10:39 +00:00
|
|
|
'You will no longer be able to hold a position on this market when it closes in {{duration}}.',
|
|
|
|
{ duration }
|
2023-10-10 15:50:49 +00:00
|
|
|
)}{' '}
|
|
|
|
{price &&
|
|
|
|
assetSymbol &&
|
2023-11-16 03:10:39 +00:00
|
|
|
t('The final price will be {{price}} {{assetSymbol}}.', {
|
|
|
|
price: addDecimalsFormatNumber(price, market.decimalPlaces),
|
2023-10-10 15:50:49 +00:00
|
|
|
assetSymbol,
|
2023-11-16 03:10:39 +00:00
|
|
|
})}
|
2023-10-10 15:50:49 +00:00
|
|
|
</div>
|
2023-10-23 11:42:37 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
} else if (openProposals.length > 1) {
|
|
|
|
content = (
|
|
|
|
<>
|
|
|
|
<div className="uppercase mb-1">
|
|
|
|
{t(
|
2023-11-16 03:10:39 +00:00
|
|
|
'Trading on Market {{name}} may stop. There are open proposals to close this market',
|
|
|
|
{ name }
|
2023-10-23 11:42:37 +00:00
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<ExternalLink href={proposalsLink}>
|
|
|
|
{t('View proposals')}
|
|
|
|
</ExternalLink>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
const { date, price } = getMessageVariables(openProposals[0]);
|
|
|
|
content = (
|
|
|
|
<>
|
|
|
|
<div className="uppercase mb-1">
|
|
|
|
{t(
|
2023-11-16 03:10:39 +00:00
|
|
|
'Trading on Market {{name}} may stop on {{date}}. There is open proposal to close this market.',
|
|
|
|
{ name, date }
|
2023-10-23 11:42:37 +00:00
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
{price &&
|
|
|
|
assetSymbol &&
|
2023-11-16 03:10:39 +00:00
|
|
|
t('Proposed final price is {{price}} {{assetSymbol}}.', {
|
|
|
|
price: addDecimalsFormatNumber(price, market.decimalPlaces),
|
2023-10-23 11:42:37 +00:00
|
|
|
assetSymbol,
|
2023-11-16 03:10:39 +00:00
|
|
|
})}
|
2023-10-23 11:42:37 +00:00
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<ExternalLink href={proposalLink}>{t('View proposal')}</ExternalLink>
|
|
|
|
</div>
|
|
|
|
</>
|
2023-10-10 15:50:49 +00:00
|
|
|
);
|
|
|
|
}
|
2023-10-23 11:42:37 +00:00
|
|
|
return (
|
|
|
|
<NotificationBanner
|
|
|
|
intent={openProposals.length ? Intent.Warning : Intent.Info}
|
|
|
|
onClose={() => {
|
|
|
|
setVisible(false);
|
|
|
|
}}
|
|
|
|
data-testid={`termination-warning-banner-${market.id}`}
|
|
|
|
>
|
|
|
|
{content}
|
|
|
|
</NotificationBanner>
|
|
|
|
);
|
2023-10-10 15:50:49 +00:00
|
|
|
};
|