2024-01-02 11:04:43 +00:00
|
|
|
import { type ReactNode } from 'react';
|
2023-10-23 11:42:37 +00:00
|
|
|
import sortBy from 'lodash/sortBy';
|
2024-01-02 11:04:43 +00:00
|
|
|
import { format, formatDuration, intervalToDuration } from 'date-fns';
|
|
|
|
import { type MarketViewProposalFieldsFragment } from '@vegaprotocol/proposals';
|
|
|
|
import { ProposalState } from '@vegaprotocol/types';
|
2023-10-23 11:42:37 +00:00
|
|
|
import {
|
|
|
|
DApp,
|
|
|
|
TOKEN_PROPOSAL,
|
|
|
|
TOKEN_PROPOSALS,
|
|
|
|
useLinks,
|
|
|
|
} from '@vegaprotocol/environment';
|
2024-01-02 11:04:43 +00:00
|
|
|
import { getQuoteName, type Market } from '@vegaprotocol/markets';
|
|
|
|
import { addDecimalsFormatNumber } from '@vegaprotocol/utils';
|
2023-11-16 03:10:39 +00:00
|
|
|
import { useT } from '../../lib/use-t';
|
2024-01-02 11:04:43 +00:00
|
|
|
import { ExternalLink } from '@vegaprotocol/ui-toolkit';
|
2023-10-23 11:42:37 +00:00
|
|
|
|
2024-01-02 11:04:43 +00:00
|
|
|
export const MarketUpdateStateBanner = ({
|
2023-10-10 15:50:49 +00:00
|
|
|
market,
|
2024-01-02 11:04:43 +00:00
|
|
|
proposals,
|
2023-10-10 15:50:49 +00:00
|
|
|
}: {
|
2024-01-02 11:04:43 +00:00
|
|
|
market: Market;
|
|
|
|
proposals: MarketViewProposalFieldsFragment[];
|
2023-10-10 15:50:49 +00:00
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-10-23 11:42:37 +00:00
|
|
|
const governanceLink = useLinks(DApp.Governance);
|
|
|
|
|
2024-01-02 11:04:43 +00:00
|
|
|
const openProposals = sortBy(
|
|
|
|
proposals.filter((p) => p.state === ProposalState.STATE_OPEN),
|
|
|
|
(p) => p.terms.enactmentDatetime
|
|
|
|
);
|
|
|
|
const passedProposals = sortBy(
|
|
|
|
proposals.filter((p) => p.state === ProposalState.STATE_PASSED),
|
|
|
|
(p) => p.terms.enactmentDatetime
|
|
|
|
);
|
2023-10-10 15:50:49 +00:00
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
if (!passedProposals.length && !openProposals.length) {
|
|
|
|
return null;
|
|
|
|
}
|
2023-10-10 15:50:49 +00:00
|
|
|
|
2024-01-02 11:04:43 +00:00
|
|
|
const name = market.tradableInstrument.instrument.code;
|
2023-10-23 11:42:37 +00:00
|
|
|
const assetSymbol = getQuoteName(market);
|
2024-01-02 11:04:43 +00:00
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
const proposalLink =
|
|
|
|
!passedProposals.length && openProposals[0]?.id
|
|
|
|
? governanceLink(TOKEN_PROPOSAL.replace(':id', openProposals[0]?.id))
|
|
|
|
: undefined;
|
2024-01-02 11:04:43 +00:00
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
const proposalsLink =
|
|
|
|
openProposals.length > 1 ? governanceLink(TOKEN_PROPOSALS) : undefined;
|
2024-01-02 11:04:43 +00:00
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
let content: ReactNode;
|
2024-01-02 11:04:43 +00:00
|
|
|
|
2023-10-23 11:42:37 +00:00
|
|
|
if (passedProposals.length) {
|
|
|
|
const { date, duration, price } = getMessageVariables(passedProposals[0]);
|
|
|
|
content = (
|
|
|
|
<>
|
2024-01-02 11:04:43 +00:00
|
|
|
<p className="uppercase mb-1">
|
|
|
|
{t('Trading on market {{name}} will stop on {{date}}', {
|
2023-11-16 03:10:39 +00:00
|
|
|
name,
|
|
|
|
date,
|
|
|
|
})}
|
2024-01-02 11:04:43 +00:00
|
|
|
</p>
|
|
|
|
<p>
|
2023-10-10 15:50:49 +00:00
|
|
|
{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
|
|
|
})}
|
2024-01-02 11:04:43 +00:00
|
|
|
</p>
|
2023-10-23 11:42:37 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
} else if (openProposals.length > 1) {
|
|
|
|
content = (
|
|
|
|
<>
|
2024-01-02 11:04:43 +00:00
|
|
|
<p className="uppercase mb-1">
|
2023-10-23 11:42:37 +00:00
|
|
|
{t(
|
2024-01-02 11:04:43 +00:00
|
|
|
'Trading on market {{name}} may stop. There are open proposals to close this market',
|
2023-11-16 03:10:39 +00:00
|
|
|
{ name }
|
2023-10-23 11:42:37 +00:00
|
|
|
)}
|
2024-01-02 11:04:43 +00:00
|
|
|
</p>
|
|
|
|
<p>
|
2023-10-23 11:42:37 +00:00
|
|
|
<ExternalLink href={proposalsLink}>
|
|
|
|
{t('View proposals')}
|
|
|
|
</ExternalLink>
|
2024-01-02 11:04:43 +00:00
|
|
|
</p>
|
2023-10-23 11:42:37 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
const { date, price } = getMessageVariables(openProposals[0]);
|
|
|
|
content = (
|
|
|
|
<>
|
2024-01-02 11:04:43 +00:00
|
|
|
<p className="mb-1">
|
2023-10-23 11:42:37 +00:00
|
|
|
{t(
|
2024-01-02 11:04:43 +00:00
|
|
|
'Trading on market {{name}} may stop on {{date}}. There is an open proposal to close this market.',
|
2023-11-16 03:10:39 +00:00
|
|
|
{ name, date }
|
2023-10-23 11:42:37 +00:00
|
|
|
)}
|
2024-01-02 11:04:43 +00:00
|
|
|
</p>
|
|
|
|
<p>
|
2023-10-23 11:42:37 +00:00
|
|
|
{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,
|
2024-01-02 11:04:43 +00:00
|
|
|
})}{' '}
|
2023-10-23 11:42:37 +00:00
|
|
|
<ExternalLink href={proposalLink}>{t('View proposal')}</ExternalLink>
|
2024-01-02 11:04:43 +00:00
|
|
|
</p>
|
2023-10-23 11:42:37 +00:00
|
|
|
</>
|
2023-10-10 15:50:49 +00:00
|
|
|
);
|
|
|
|
}
|
2024-01-02 11:04:43 +00:00
|
|
|
|
|
|
|
return <div data-testid={`update-state-banner-${market.id}`}>{content}</div>;
|
|
|
|
};
|
|
|
|
|
|
|
|
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'],
|
|
|
|
}
|
2023-10-23 11:42:37 +00:00
|
|
|
);
|
2024-01-02 11:04:43 +00:00
|
|
|
const price =
|
|
|
|
proposal.terms.change.__typename === 'UpdateMarketState'
|
|
|
|
? proposal.terms.change.price
|
|
|
|
: '';
|
|
|
|
return {
|
|
|
|
date,
|
|
|
|
duration,
|
|
|
|
price,
|
|
|
|
};
|
2023-10-10 15:50:49 +00:00
|
|
|
};
|