import { useActiveRewardsQuery } from './__generated__/Rewards'; import { useT } from '../../lib/use-t'; import { addDecimalsFormatNumber } from '@vegaprotocol/utils'; import classNames from 'classnames'; import { type IconName, type VegaIconSize, Icon, Intent, Tooltip, VegaIcon, VegaIconNames, TradingInput, TinyScroll, truncateMiddle, } from '@vegaprotocol/ui-toolkit'; import { IconNames } from '@blueprintjs/icons'; import { type Maybe, type Transfer, type TransferNode, type RecurringTransfer, DistributionStrategyDescriptionMapping, DistributionStrategyMapping, EntityScope, EntityScopeMapping, TransferStatus, TransferStatusMapping, DispatchMetric, DispatchMetricDescription, DispatchMetricLabels, EntityScopeLabelMapping, MarketState, type DispatchStrategy, IndividualScopeMapping, IndividualScopeDescriptionMapping, } from '@vegaprotocol/types'; import { Card } from '../card/card'; import { useMemo, useState } from 'react'; import { type AssetFieldsFragment, useAssetsMapProvider, } from '@vegaprotocol/assets'; import { type MarketFieldsFragment, useMarketsMapProvider, getAsset, } from '@vegaprotocol/markets'; export type Filter = { searchTerm: string; }; export const isActiveReward = (node: TransferNode, currentEpoch: number) => { const { transfer } = node; if (transfer.kind.__typename !== 'RecurringTransfer') { return false; } const { dispatchStrategy } = transfer.kind; if (!dispatchStrategy) { return false; } if (transfer.kind.endEpoch && transfer.kind.endEpoch < currentEpoch) { return false; } if (transfer.status !== TransferStatus.STATUS_PENDING) { return false; } return true; }; export const applyFilter = ( node: TransferNode & { asset?: AssetFieldsFragment | null; markets?: (MarketFieldsFragment | null)[]; }, filter: Filter ) => { const { transfer } = node; if ( transfer.kind.__typename !== 'RecurringTransfer' || !transfer.kind.dispatchStrategy?.dispatchMetric ) { return false; } if ( DispatchMetricLabels[transfer.kind.dispatchStrategy.dispatchMetric] .toLowerCase() .includes(filter.searchTerm.toLowerCase()) || transfer.asset?.symbol .toLowerCase() .includes(filter.searchTerm.toLowerCase()) || EntityScopeLabelMapping[transfer.kind.dispatchStrategy.entityScope] .toLowerCase() .includes(filter.searchTerm.toLowerCase()) || node.asset?.name .toLocaleLowerCase() .includes(filter.searchTerm.toLowerCase()) || node.markets?.some((m) => m?.tradableInstrument?.instrument?.name .toLocaleLowerCase() .includes(filter.searchTerm.toLowerCase()) ) ) { return true; } return false; }; export const ActiveRewards = ({ currentEpoch }: { currentEpoch: number }) => { const t = useT(); const { data: activeRewardsData } = useActiveRewardsQuery({ variables: { isReward: true, }, }); const [filter, setFilter] = useState({ searchTerm: '', }); const { data: assets } = useAssetsMapProvider(); const { data: markets } = useMarketsMapProvider(); const enrichedTransfers = activeRewardsData?.transfersConnection?.edges ?.map((e) => e?.node as TransferNode) .filter((node) => isActiveReward(node, currentEpoch)) .map((node) => { if (node.transfer.kind.__typename !== 'RecurringTransfer') { return node; } const asset = assets && assets[ node.transfer.kind.dispatchStrategy?.dispatchMetricAssetId || '' ]; const marketsInScope = node.transfer.kind.dispatchStrategy?.marketIdsInScope?.map( (id) => markets && markets[id] ); return { ...node, asset, markets: marketsInScope }; }); if (!enrichedTransfers || !enrichedTransfers.length) return null; return ( {enrichedTransfers.length > 1 && ( setFilter((curr) => ({ ...curr, searchTerm: e.target.value })) } value={filter.searchTerm} type="text" placeholder={t( 'Search by reward dispatch metric, entity scope or asset name' )} data-testid="search-term" className="mb-4 w-20 mr-2" prependElement={} /> )} {enrichedTransfers .filter((n) => applyFilter(n, filter)) .map((node, i) => { const { transfer } = node; if ( transfer.kind.__typename !== 'RecurringTransfer' || !transfer.kind.dispatchStrategy?.dispatchMetric ) { return null; } return ( node && ( ) ); })} ); }; // This was built to be a status indicator for the rewards based on the transfer status // eslint-disable-next-line @typescript-eslint/no-unused-vars const StatusIndicator = ({ status, reason, }: { status: TransferStatus; reason?: Maybe | undefined; }) => { const t = useT(); const getIconIntent = (status: string) => { switch (status) { case TransferStatus.STATUS_DONE: return { icon: IconNames.TICK_CIRCLE, intent: Intent.Success }; case TransferStatus.STATUS_REJECTED: return { icon: IconNames.ERROR, intent: Intent.Danger }; default: return { icon: IconNames.HELP, intent: Intent.Primary }; } }; const { icon, intent } = getIconIntent(status); return ( {t('Transfer status: {{status}} {{reason}}', { status: TransferStatusMapping[status], reason: reason ? `(${reason})` : '', })} } > ); }; export const ActiveRewardCard = ({ transferNode, currentEpoch, kind, allMarkets, }: { transferNode: TransferNode & { asset?: AssetFieldsFragment | null; markets?: (MarketFieldsFragment | null)[]; }; currentEpoch: number; kind: RecurringTransfer; allMarkets?: Record; }) => { const t = useT(); const { transfer } = transferNode; const { dispatchStrategy } = kind; const marketIdsInScope = dispatchStrategy?.marketIdsInScope; const firstMarketData = transferNode.markets?.[0]; const specificMarkets = useMemo(() => { if ( !firstMarketData || !marketIdsInScope || marketIdsInScope.length === 0 ) { return null; } if (marketIdsInScope.length > 1) { const marketNames = allMarkets && marketIdsInScope .map((id) => allMarkets[id]?.tradableInstrument?.instrument?.name) .join(', '); return ( Specific markets ); } return ( {firstMarketData?.tradableInstrument?.instrument?.name || ''} ); }, [firstMarketData, marketIdsInScope, allMarkets]); const dispatchAsset = transferNode.asset; if (!dispatchStrategy) { return null; } // Gray out/hide the cards that are related to not trading markets const marketSettled = transferNode.markets?.some( (m) => m?.state && [ MarketState.STATE_TRADING_TERMINATED, MarketState.STATE_SETTLED, MarketState.STATE_CANCELLED, MarketState.STATE_CLOSED, ].includes(m.state) ); if (marketSettled) { return null; } const assetInSettledMarket = allMarkets && Object.values(allMarkets).some((m: MarketFieldsFragment | null) => { if (m && getAsset(m).id === dispatchStrategy.dispatchMetricAssetId) { return ( m?.state && [ MarketState.STATE_TRADING_TERMINATED, MarketState.STATE_SETTLED, MarketState.STATE_CANCELLED, MarketState.STATE_CLOSED, ].includes(m.state) ); } return false; }); // Gray out the cards that are related to suspended markets const suspended = transferNode.markets?.some( (m) => m?.state === MarketState.STATE_SUSPENDED || m?.state === MarketState.STATE_SUSPENDED_VIA_GOVERNANCE ); const assetInSuspendedMarket = allMarkets && Object.values(allMarkets).some((m: MarketFieldsFragment | null) => { if (m && getAsset(m).id === dispatchStrategy.dispatchMetricAssetId) { return ( m?.state === MarketState.STATE_SUSPENDED || m?.state === MarketState.STATE_SUSPENDED_VIA_GOVERNANCE ); } return false; }); // Gray out the cards that are related to suspended markets const { gradientClassName, mainClassName } = suspended || assetInSuspendedMarket || assetInSettledMarket ? { gradientClassName: 'from-vega-cdark-500 to-vega-clight-400', mainClassName: 'from-vega-cdark-400 dark:from-vega-cdark-600 to-20%', } : getGradientClasses(dispatchStrategy.dispatchMetric); const entityScope = dispatchStrategy.entityScope; return (
{entityScope && ( {EntityScopeLabelMapping[entityScope] || t('Unspecified')} )}

{addDecimalsFormatNumber( transferNode.transfer.amount, transferNode.transfer.asset?.decimals || 0, 6 )} {transferNode.transfer.asset?.symbol}

{ { DistributionStrategyMapping[ dispatchStrategy.distributionStrategy ] } }
{t('numberEpochs', '{{count}} epochs', { count: kind.dispatchStrategy?.lockPeriod, })}
{DispatchMetricLabels[dispatchStrategy.dispatchMetric]} •{' '} {specificMarkets || dispatchAsset?.name}
{kind.endEpoch && ( {t('Ends in')} {t('numberEpochs', '{{count}} epochs', { count: kind.endEpoch - currentEpoch, })} )} { {t('Assessed over')} {t('numberEpochs', '{{count}} epochs', { count: dispatchStrategy.windowLength, })} }
{dispatchStrategy?.dispatchMetric && ( {t(DispatchMetricDescription[dispatchStrategy?.dispatchMetric])} )} {kind.dispatchStrategy && ( )}
); }; const RewardRequirements = ({ dispatchStrategy, assetDecimalPlaces = 0, }: { dispatchStrategy: DispatchStrategy; assetDecimalPlaces: number | undefined; }) => { const t = useT(); return (
{t('{{entity}} scope', { entity: EntityScopeLabelMapping[dispatchStrategy.entityScope], })}
{t('Staked VEGA')}
{addDecimalsFormatNumber( dispatchStrategy?.stakingRequirement || 0, assetDecimalPlaces )}
{t('Average position')}
{addDecimalsFormatNumber( dispatchStrategy?.notionalTimeWeightedAveragePositionRequirement || 0, assetDecimalPlaces )}
); }; const RewardEntityScope = ({ dispatchStrategy, }: { dispatchStrategy: DispatchStrategy; }) => { const t = useT(); if (dispatchStrategy.entityScope === EntityScope.ENTITY_SCOPE_TEAMS) { return (

{t('Eligible teams')}

    {dispatchStrategy.teamScope.map((teamId) => { if (!teamId) return null; return
  • {truncateMiddle(teamId)}
  • ; })}
) : ( t('All teams are eligible') ) } > {dispatchStrategy.teamScope?.length ? t('Some teams') : t('All teams')}
); } if ( dispatchStrategy.entityScope === EntityScope.ENTITY_SCOPE_INDIVIDUALS && dispatchStrategy.individualScope ) { return ( {IndividualScopeMapping[dispatchStrategy.individualScope]} ); } return null; }; const getGradientClasses = (d: DispatchMetric | undefined) => { switch (d) { case DispatchMetric.DISPATCH_METRIC_AVERAGE_POSITION: return { gradientClassName: 'from-vega-pink-500 to-vega-purple-400', mainClassName: 'from-vega-pink-400 dark:from-vega-pink-600 to-20%', }; case DispatchMetric.DISPATCH_METRIC_LP_FEES_RECEIVED: return { gradientClassName: 'from-vega-green-500 to-vega-yellow-500', mainClassName: 'from-vega-green-400 dark:from-vega-green-600 to-20%', }; case DispatchMetric.DISPATCH_METRIC_MAKER_FEES_PAID: return { gradientClassName: 'from-vega-orange-500 to-vega-pink-400', mainClassName: 'from-vega-orange-400 dark:from-vega-orange-600 to-20%', }; case DispatchMetric.DISPATCH_METRIC_MARKET_VALUE: case DispatchMetric.DISPATCH_METRIC_RELATIVE_RETURN: return { gradientClassName: 'from-vega-purple-500 to-vega-blue-400', mainClassName: 'from-vega-purple-400 dark:from-vega-purple-600 to-20%', }; case DispatchMetric.DISPATCH_METRIC_RETURN_VOLATILITY: return { gradientClassName: 'from-vega-blue-500 to-vega-green-400', mainClassName: 'from-vega-blue-400 dark:from-vega-blue-600 to-20%', }; case DispatchMetric.DISPATCH_METRIC_VALIDATOR_RANKING: default: return { gradientClassName: 'from-vega-pink-500 to-vega-purple-400', mainClassName: 'from-vega-pink-400 dark:from-vega-pink-600 to-20%', }; } }; const CardIcon = ({ size = 18, iconName, tooltip, }: { size?: VegaIconSize; iconName: VegaIconNames; tooltip: string; }) => { return ( {tooltip}}> ); }; const EntityIcon = ({ transfer, size = 18, }: { transfer: Transfer; size?: VegaIconSize; }) => { if (transfer.kind.__typename !== 'RecurringTransfer') { return null; } const entityScope = transfer.kind.dispatchStrategy?.entityScope; const getIconName = () => { switch (entityScope) { case EntityScope.ENTITY_SCOPE_TEAMS: return VegaIconNames.TEAM; case EntityScope.ENTITY_SCOPE_INDIVIDUALS: return VegaIconNames.MAN; default: return VegaIconNames.QUESTION_MARK; } }; const iconName = getIconName(); return ( {entityScope ? EntityScopeMapping[entityScope] : ''} } > {iconName && } ); };