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';
|
2024-02-26 16:17:08 +00:00
|
|
|
import { type ProposalFragment } from '@vegaprotocol/proposals';
|
|
|
|
import { MarketUpdateType, 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;
|
2024-02-26 16:17:08 +00:00
|
|
|
proposals: ProposalFragment[];
|
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-02-26 16:17:08 +00:00
|
|
|
const openTradingProposals = sortBy(
|
|
|
|
proposals.filter(
|
|
|
|
(p) =>
|
|
|
|
p.terms &&
|
|
|
|
p.terms.change.__typename === 'UpdateMarketState' &&
|
|
|
|
p.terms.change.updateType ===
|
|
|
|
MarketUpdateType.MARKET_STATE_UPDATE_TYPE_RESUME
|
|
|
|
),
|
|
|
|
(p) => p.terms?.enactmentDatetime
|
|
|
|
);
|
|
|
|
|
2024-01-02 11:04:43 +00:00
|
|
|
const openProposals = sortBy(
|
|
|
|
proposals.filter((p) => p.state === ProposalState.STATE_OPEN),
|
2024-02-26 16:17:08 +00:00
|
|
|
(p) => p.terms?.enactmentDatetime
|
2024-01-02 11:04:43 +00:00
|
|
|
);
|
|
|
|
const passedProposals = sortBy(
|
|
|
|
proposals.filter((p) => p.state === ProposalState.STATE_PASSED),
|
2024-02-26 16:17:08 +00:00
|
|
|
(p) => p.terms?.enactmentDatetime
|
2024-01-02 11:04:43 +00:00
|
|
|
);
|
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
|
|
|
|
2024-02-26 16:17:08 +00:00
|
|
|
const openTradingProposalsLink =
|
|
|
|
openTradingProposals[0]?.__typename === 'Proposal' &&
|
|
|
|
openTradingProposals[0]?.id
|
|
|
|
? governanceLink(
|
|
|
|
TOKEN_PROPOSAL.replace(':id', openTradingProposals[0].id)
|
|
|
|
)
|
|
|
|
: openTradingProposals[0]?.__typename === 'ProposalDetail' &&
|
|
|
|
openTradingProposals[0]?.batchId
|
|
|
|
? governanceLink(
|
|
|
|
TOKEN_PROPOSAL.replace(':id', openTradingProposals[0].batchId)
|
|
|
|
)
|
|
|
|
: undefined;
|
|
|
|
|
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
|
|
|
|
2024-02-26 16:17:08 +00:00
|
|
|
if (openTradingProposals.length >= 1) {
|
|
|
|
content = (
|
|
|
|
<>
|
|
|
|
<p className="mb-1">
|
|
|
|
{t(
|
|
|
|
'Trading on market {{name}} was suspended by governance. There are open proposals to resume trading on this market.',
|
|
|
|
{ name }
|
|
|
|
)}
|
|
|
|
</p>
|
|
|
|
<p>
|
|
|
|
<ExternalLink href={openTradingProposalsLink}>
|
|
|
|
{t('View proposals')}
|
|
|
|
</ExternalLink>
|
|
|
|
</p>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
} else if (passedProposals.length) {
|
2023-10-23 11:42:37 +00:00
|
|
|
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>;
|
|
|
|
};
|
|
|
|
|
2024-02-26 16:17:08 +00:00
|
|
|
const getMessageVariables = (proposal: ProposalFragment) => {
|
|
|
|
const enactmentDatetime =
|
|
|
|
proposal.terms && new Date(proposal.terms.enactmentDatetime);
|
|
|
|
const date = enactmentDatetime && format(enactmentDatetime, 'dd MMMM');
|
|
|
|
const duration =
|
|
|
|
enactmentDatetime &&
|
|
|
|
formatDuration(
|
|
|
|
intervalToDuration({
|
|
|
|
start: new Date(),
|
|
|
|
end: enactmentDatetime,
|
|
|
|
}),
|
|
|
|
{
|
|
|
|
format: ['days', 'hours'],
|
|
|
|
}
|
|
|
|
);
|
2024-01-02 11:04:43 +00:00
|
|
|
const price =
|
2024-02-26 16:17:08 +00:00
|
|
|
proposal.terms?.change.__typename === 'UpdateMarketState'
|
2024-01-02 11:04:43 +00:00
|
|
|
? proposal.terms.change.price
|
|
|
|
: '';
|
|
|
|
return {
|
|
|
|
date,
|
|
|
|
duration,
|
|
|
|
price,
|
|
|
|
};
|
2023-10-10 15:50:49 +00:00
|
|
|
};
|