2023-10-25 21:59:30 +00:00
|
|
|
import maxBy from 'lodash/maxBy';
|
|
|
|
import minBy from 'lodash/minBy';
|
2024-03-01 14:25:56 +00:00
|
|
|
import { useVegaWallet } from '@vegaprotocol/wallet-react';
|
2023-10-25 21:59:30 +00:00
|
|
|
import {
|
|
|
|
useNetworkParams,
|
|
|
|
NetworkParams,
|
|
|
|
} from '@vegaprotocol/network-parameters';
|
|
|
|
import { useMarketList } from '@vegaprotocol/markets';
|
2023-10-27 13:20:45 +00:00
|
|
|
import { formatNumber, formatNumberRounded } from '@vegaprotocol/utils';
|
2023-10-25 21:59:30 +00:00
|
|
|
import { useDiscountProgramsQuery, useFeesQuery } from './__generated__/Fees';
|
2023-11-15 21:46:19 +00:00
|
|
|
import { Card, CardStat, CardTable, CardTableTD, CardTableTH } from '../card';
|
2023-10-25 21:59:30 +00:00
|
|
|
import { MarketFees } from './market-fees';
|
|
|
|
import { useVolumeStats } from './use-volume-stats';
|
|
|
|
import { useReferralStats } from './use-referral-stats';
|
|
|
|
import { formatPercentage, getAdjustedFee } from './utils';
|
2024-01-02 15:50:04 +00:00
|
|
|
import { Table as SimpleTable } from '../../components/table';
|
2023-10-25 21:59:30 +00:00
|
|
|
import BigNumber from 'bignumber.js';
|
2023-11-09 15:50:50 +00:00
|
|
|
import { Links } from '../../lib/links';
|
|
|
|
import { Link } from 'react-router-dom';
|
|
|
|
import {
|
|
|
|
Tooltip,
|
|
|
|
VegaIcon,
|
|
|
|
VegaIconNames,
|
|
|
|
truncateMiddle,
|
|
|
|
} from '@vegaprotocol/ui-toolkit';
|
2023-11-16 03:10:39 +00:00
|
|
|
import { useT } from '../../lib/use-t';
|
2024-01-02 15:50:04 +00:00
|
|
|
import classNames from 'classnames';
|
|
|
|
import { getTierGradient } from '../helpers/tiers';
|
2023-10-25 21:59:30 +00:00
|
|
|
|
|
|
|
export const FeesContainer = () => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-10-25 21:59:30 +00:00
|
|
|
const { pubKey } = useVegaWallet();
|
|
|
|
const { params, loading: paramsLoading } = useNetworkParams([
|
|
|
|
NetworkParams.market_fee_factors_makerFee,
|
|
|
|
NetworkParams.market_fee_factors_infrastructureFee,
|
|
|
|
]);
|
|
|
|
|
|
|
|
const { data: markets, loading: marketsLoading } = useMarketList();
|
|
|
|
|
|
|
|
const { data: programData, loading: programLoading } =
|
2024-01-05 12:43:30 +00:00
|
|
|
useDiscountProgramsQuery({
|
|
|
|
errorPolicy: 'ignore',
|
|
|
|
fetchPolicy: 'cache-and-network',
|
|
|
|
pollInterval: 15000,
|
|
|
|
});
|
2023-10-25 21:59:30 +00:00
|
|
|
|
2023-11-03 15:11:51 +00:00
|
|
|
const volumeDiscountWindowLength =
|
2023-10-25 21:59:30 +00:00
|
|
|
programData?.currentVolumeDiscountProgram?.windowLength || 1;
|
2023-11-03 15:11:51 +00:00
|
|
|
const referralDiscountWindowLength =
|
2023-10-25 21:59:30 +00:00
|
|
|
programData?.currentReferralProgram?.windowLength || 1;
|
|
|
|
const { data: feesData, loading: feesLoading } = useFeesQuery({
|
|
|
|
variables: {
|
|
|
|
partyId: pubKey || '',
|
|
|
|
},
|
2023-12-01 17:03:41 +00:00
|
|
|
skip: !pubKey,
|
2024-01-05 12:43:30 +00:00
|
|
|
fetchPolicy: 'cache-and-network',
|
|
|
|
pollInterval: 15000,
|
2023-10-25 21:59:30 +00:00
|
|
|
});
|
|
|
|
|
2023-12-01 17:03:41 +00:00
|
|
|
const previousEpoch = (Number(feesData?.epoch.id) || 0) - 1;
|
|
|
|
|
2023-10-25 21:59:30 +00:00
|
|
|
const { volumeDiscount, volumeTierIndex, volumeInWindow, volumeTiers } =
|
|
|
|
useVolumeStats(
|
2023-12-01 17:03:41 +00:00
|
|
|
previousEpoch,
|
|
|
|
feesData?.volumeDiscountStats.edges?.[0]?.node,
|
2023-10-25 21:59:30 +00:00
|
|
|
programData?.currentVolumeDiscountProgram
|
|
|
|
);
|
|
|
|
|
|
|
|
const {
|
|
|
|
referralDiscount,
|
|
|
|
referralVolumeInWindow,
|
|
|
|
referralTierIndex,
|
|
|
|
referralTiers,
|
|
|
|
epochsInSet,
|
2023-11-09 15:50:50 +00:00
|
|
|
code,
|
|
|
|
isReferrer,
|
2023-10-25 21:59:30 +00:00
|
|
|
} = useReferralStats(
|
2023-12-01 17:03:41 +00:00
|
|
|
previousEpoch,
|
|
|
|
feesData?.referralSetStats.edges?.[0]?.node,
|
|
|
|
feesData?.referralSetReferees.edges?.[0]?.node,
|
2023-10-25 21:59:30 +00:00
|
|
|
programData?.currentReferralProgram,
|
2023-12-01 17:03:41 +00:00
|
|
|
feesData?.referrer.edges?.[0]?.node,
|
|
|
|
feesData?.referee.edges?.[0]?.node
|
2023-10-25 21:59:30 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const loading = paramsLoading || feesLoading || programLoading;
|
|
|
|
const isConnected = Boolean(pubKey);
|
|
|
|
|
2023-11-09 15:50:50 +00:00
|
|
|
const isReferralProgramRunning = Boolean(programData?.currentReferralProgram);
|
|
|
|
const isVolumeDiscountProgramRunning = Boolean(
|
|
|
|
programData?.currentVolumeDiscountProgram
|
|
|
|
);
|
|
|
|
|
2023-10-25 21:59:30 +00:00
|
|
|
return (
|
2023-12-19 12:22:15 +00:00
|
|
|
<div
|
|
|
|
className="grid auto-rows-min grid-cols-4 gap-3"
|
|
|
|
data-testid="fees-container"
|
|
|
|
>
|
2023-10-25 21:59:30 +00:00
|
|
|
{isConnected && (
|
|
|
|
<>
|
2023-11-15 21:46:19 +00:00
|
|
|
<Card
|
2023-10-25 21:59:30 +00:00
|
|
|
title={t('My trading fees')}
|
|
|
|
className="sm:col-span-2"
|
|
|
|
loading={loading}
|
|
|
|
>
|
|
|
|
<TradingFees
|
|
|
|
params={params}
|
|
|
|
markets={markets}
|
|
|
|
referralDiscount={referralDiscount}
|
|
|
|
volumeDiscount={volumeDiscount}
|
|
|
|
/>
|
2023-11-15 21:46:19 +00:00
|
|
|
</Card>
|
|
|
|
<Card
|
2023-10-25 21:59:30 +00:00
|
|
|
title={t('Total discount')}
|
|
|
|
className="sm:col-span-2"
|
|
|
|
loading={loading}
|
|
|
|
>
|
|
|
|
<TotalDiscount
|
|
|
|
referralDiscount={referralDiscount}
|
|
|
|
volumeDiscount={volumeDiscount}
|
2023-11-09 15:50:50 +00:00
|
|
|
isReferralProgramRunning={isReferralProgramRunning}
|
|
|
|
isVolumeDiscountProgramRunning={isVolumeDiscountProgramRunning}
|
2023-10-25 21:59:30 +00:00
|
|
|
/>
|
2023-11-15 21:46:19 +00:00
|
|
|
</Card>
|
|
|
|
<Card
|
2023-10-25 21:59:30 +00:00
|
|
|
title={t('My current volume')}
|
|
|
|
className="sm:col-span-2"
|
|
|
|
loading={loading}
|
|
|
|
>
|
2023-11-09 15:50:50 +00:00
|
|
|
{isVolumeDiscountProgramRunning ? (
|
|
|
|
<CurrentVolume
|
|
|
|
tiers={volumeTiers}
|
|
|
|
tierIndex={volumeTierIndex}
|
|
|
|
windowLengthVolume={volumeInWindow}
|
|
|
|
windowLength={volumeDiscountWindowLength}
|
|
|
|
/>
|
|
|
|
) : (
|
2023-12-19 12:22:15 +00:00
|
|
|
<p
|
|
|
|
className="text-muted pt-3 text-sm"
|
|
|
|
data-testid="no-volume-discount"
|
|
|
|
>
|
2023-11-09 15:50:50 +00:00
|
|
|
{t('No volume discount program active')}
|
|
|
|
</p>
|
|
|
|
)}
|
2023-11-15 21:46:19 +00:00
|
|
|
</Card>
|
|
|
|
<Card
|
2023-10-25 21:59:30 +00:00
|
|
|
title={t('Referral benefits')}
|
|
|
|
className="sm:col-span-2"
|
|
|
|
loading={loading}
|
2023-12-19 12:22:15 +00:00
|
|
|
data-testid="referral-benefits-card"
|
2023-10-25 21:59:30 +00:00
|
|
|
>
|
2023-11-09 15:50:50 +00:00
|
|
|
{isReferrer ? (
|
2023-12-19 12:22:15 +00:00
|
|
|
<ReferrerInfo code={code} data-testid="referrer-info" />
|
2023-11-09 15:50:50 +00:00
|
|
|
) : isReferralProgramRunning ? (
|
|
|
|
<ReferralBenefits
|
|
|
|
setRunningNotionalTakerVolume={referralVolumeInWindow}
|
|
|
|
epochsInSet={epochsInSet}
|
|
|
|
epochs={referralDiscountWindowLength}
|
2023-12-19 12:22:15 +00:00
|
|
|
data-testid="referral-benefits"
|
2023-11-09 15:50:50 +00:00
|
|
|
/>
|
|
|
|
) : (
|
2023-12-19 12:22:15 +00:00
|
|
|
<p
|
|
|
|
className="text-muted pt-3 text-sm"
|
|
|
|
data-testid="no-referral-program"
|
|
|
|
>
|
2023-11-09 15:50:50 +00:00
|
|
|
{t('No referral program active')}
|
|
|
|
</p>
|
|
|
|
)}
|
2023-11-15 21:46:19 +00:00
|
|
|
</Card>
|
2023-10-25 21:59:30 +00:00
|
|
|
</>
|
|
|
|
)}
|
2023-11-15 21:46:19 +00:00
|
|
|
<Card
|
2023-10-25 21:59:30 +00:00
|
|
|
title={t('Volume discount')}
|
|
|
|
className="lg:col-span-full xl:col-span-2"
|
|
|
|
loading={loading}
|
2023-12-19 12:22:15 +00:00
|
|
|
data-testid="volume-discount-card"
|
2024-01-02 15:50:04 +00:00
|
|
|
noBackgroundOnMobile={true}
|
2023-10-25 21:59:30 +00:00
|
|
|
>
|
|
|
|
<VolumeTiers
|
|
|
|
tiers={volumeTiers}
|
|
|
|
tierIndex={volumeTierIndex}
|
|
|
|
lastEpochVolume={volumeInWindow}
|
2023-11-03 15:11:51 +00:00
|
|
|
windowLength={volumeDiscountWindowLength}
|
2023-10-25 21:59:30 +00:00
|
|
|
/>
|
2023-11-15 21:46:19 +00:00
|
|
|
</Card>
|
|
|
|
<Card
|
2023-10-25 21:59:30 +00:00
|
|
|
title={t('Referral discount')}
|
|
|
|
className="lg:col-span-full xl:col-span-2"
|
|
|
|
loading={loading}
|
2023-12-19 12:22:15 +00:00
|
|
|
data-testid="referral-discount-card"
|
2024-01-02 15:50:04 +00:00
|
|
|
noBackgroundOnMobile={true}
|
2023-10-25 21:59:30 +00:00
|
|
|
>
|
|
|
|
<ReferralTiers
|
|
|
|
tiers={referralTiers}
|
|
|
|
tierIndex={referralTierIndex}
|
|
|
|
epochsInSet={epochsInSet}
|
|
|
|
referralVolumeInWindow={referralVolumeInWindow}
|
2024-01-02 15:50:04 +00:00
|
|
|
referralDiscountWindowLength={referralDiscountWindowLength}
|
2023-10-25 21:59:30 +00:00
|
|
|
/>
|
2023-11-15 21:46:19 +00:00
|
|
|
</Card>
|
|
|
|
<Card
|
2023-11-09 15:50:50 +00:00
|
|
|
title={t('Fees by market')}
|
2023-10-25 21:59:30 +00:00
|
|
|
className="lg:col-span-full"
|
|
|
|
loading={marketsLoading}
|
2023-12-19 12:22:15 +00:00
|
|
|
data-testid="fees-by-market-card"
|
2024-01-02 15:50:04 +00:00
|
|
|
noBackgroundOnMobile={true}
|
2023-10-25 21:59:30 +00:00
|
|
|
>
|
|
|
|
<MarketFees
|
|
|
|
markets={markets}
|
|
|
|
referralDiscount={referralDiscount}
|
|
|
|
volumeDiscount={volumeDiscount}
|
|
|
|
/>
|
2023-11-15 21:46:19 +00:00
|
|
|
</Card>
|
2023-10-25 21:59:30 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const TradingFees = ({
|
|
|
|
params,
|
|
|
|
markets,
|
|
|
|
referralDiscount,
|
|
|
|
volumeDiscount,
|
|
|
|
}: {
|
|
|
|
params: {
|
|
|
|
market_fee_factors_infrastructureFee: string;
|
|
|
|
market_fee_factors_makerFee: string;
|
|
|
|
};
|
|
|
|
markets: Array<{ fees: { factors: { liquidityFee: string } } }> | null;
|
|
|
|
referralDiscount: number;
|
|
|
|
volumeDiscount: number;
|
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-10-25 21:59:30 +00:00
|
|
|
const referralDiscountBigNum = new BigNumber(referralDiscount);
|
|
|
|
const volumeDiscountBigNum = new BigNumber(volumeDiscount);
|
|
|
|
|
|
|
|
// Show min and max liquidity fees from all markets
|
|
|
|
const minLiq = minBy(markets, (m) => Number(m.fees.factors.liquidityFee));
|
|
|
|
const maxLiq = maxBy(markets, (m) => Number(m.fees.factors.liquidityFee));
|
|
|
|
|
|
|
|
const total = new BigNumber(params.market_fee_factors_makerFee).plus(
|
|
|
|
new BigNumber(params.market_fee_factors_infrastructureFee)
|
|
|
|
);
|
|
|
|
|
|
|
|
const adjustedTotal = getAdjustedFee(
|
|
|
|
[total],
|
|
|
|
[referralDiscountBigNum, volumeDiscountBigNum]
|
|
|
|
);
|
|
|
|
|
|
|
|
let minTotal;
|
|
|
|
let maxTotal;
|
|
|
|
|
|
|
|
let minAdjustedTotal;
|
|
|
|
let maxAdjustedTotal;
|
|
|
|
|
|
|
|
if (minLiq && maxLiq) {
|
|
|
|
const minLiqFee = new BigNumber(minLiq.fees.factors.liquidityFee);
|
|
|
|
const maxLiqFee = new BigNumber(maxLiq.fees.factors.liquidityFee);
|
|
|
|
|
|
|
|
minTotal = total.plus(minLiqFee);
|
|
|
|
maxTotal = total.plus(maxLiqFee);
|
|
|
|
|
|
|
|
minAdjustedTotal = getAdjustedFee(
|
|
|
|
[total, minLiqFee],
|
|
|
|
[referralDiscountBigNum, volumeDiscountBigNum]
|
|
|
|
);
|
|
|
|
|
|
|
|
maxAdjustedTotal = getAdjustedFee(
|
|
|
|
[total, maxLiqFee],
|
|
|
|
[referralDiscountBigNum, volumeDiscountBigNum]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
2023-12-19 12:22:15 +00:00
|
|
|
<div className="pt-4" data-testid="trading-fees">
|
2023-11-15 21:46:19 +00:00
|
|
|
<div className="leading-none">
|
2023-10-25 21:59:30 +00:00
|
|
|
<p className="block text-3xl leading-none" data-testid="adjusted-fees">
|
|
|
|
{minAdjustedTotal !== undefined && maxAdjustedTotal !== undefined
|
|
|
|
? `${formatPercentage(minAdjustedTotal)}%-${formatPercentage(
|
|
|
|
maxAdjustedTotal
|
|
|
|
)}%`
|
|
|
|
: `${formatPercentage(adjustedTotal)}%`}
|
|
|
|
</p>
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTable>
|
2023-12-19 12:22:15 +00:00
|
|
|
<tr className="text-default" data-testid="total-fee-before-discount">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTableTH>{t('Total fee before discount')}</CardTableTH>
|
|
|
|
<CardTableTD>
|
|
|
|
{minTotal !== undefined && maxTotal !== undefined
|
|
|
|
? `${formatPercentage(minTotal.toNumber())}%-${formatPercentage(
|
|
|
|
maxTotal.toNumber()
|
|
|
|
)}%`
|
|
|
|
: `${formatPercentage(total.toNumber())}%`}
|
|
|
|
</CardTableTD>
|
|
|
|
</tr>
|
2023-12-19 12:22:15 +00:00
|
|
|
<tr data-testid="infrastructure-fees">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTableTH>{t('Infrastructure')}</CardTableTH>
|
|
|
|
<CardTableTD>
|
|
|
|
{formatPercentage(
|
|
|
|
Number(params.market_fee_factors_infrastructureFee)
|
|
|
|
)}
|
|
|
|
%
|
|
|
|
</CardTableTD>
|
|
|
|
</tr>
|
2023-12-19 12:22:15 +00:00
|
|
|
<tr data-testid="maker-fees">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTableTH>{t('Maker')}</CardTableTH>
|
|
|
|
<CardTableTD>
|
|
|
|
{formatPercentage(Number(params.market_fee_factors_makerFee))}%
|
|
|
|
</CardTableTD>
|
|
|
|
</tr>
|
|
|
|
{minLiq && maxLiq && (
|
2023-12-19 12:22:15 +00:00
|
|
|
<tr data-testid="liquidity-fees">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTableTH>{t('Liquidity')}</CardTableTH>
|
|
|
|
<CardTableTD>
|
|
|
|
{formatPercentage(Number(minLiq.fees.factors.liquidityFee))}%
|
|
|
|
{'-'}
|
|
|
|
{formatPercentage(Number(maxLiq.fees.factors.liquidityFee))}%
|
|
|
|
</CardTableTD>
|
2023-10-25 21:59:30 +00:00
|
|
|
</tr>
|
2023-11-15 21:46:19 +00:00
|
|
|
)}
|
|
|
|
</CardTable>
|
2023-10-25 21:59:30 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const CurrentVolume = ({
|
|
|
|
tiers,
|
|
|
|
tierIndex,
|
|
|
|
windowLengthVolume,
|
2023-11-03 15:11:51 +00:00
|
|
|
windowLength,
|
2023-10-25 21:59:30 +00:00
|
|
|
}: {
|
|
|
|
tiers: Array<{ minimumRunningNotionalTakerVolume: string }>;
|
|
|
|
tierIndex: number;
|
|
|
|
windowLengthVolume: number;
|
2023-11-03 15:11:51 +00:00
|
|
|
windowLength: number;
|
2023-10-25 21:59:30 +00:00
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-10-25 21:59:30 +00:00
|
|
|
const nextTier = tiers[tierIndex + 1];
|
|
|
|
const requiredForNextTier = nextTier
|
2023-12-05 19:36:25 +00:00
|
|
|
? new BigNumber(nextTier.minimumRunningNotionalTakerVolume).minus(
|
|
|
|
windowLengthVolume
|
|
|
|
)
|
|
|
|
: new BigNumber(0);
|
|
|
|
const currentVolume = new BigNumber(windowLengthVolume);
|
2023-10-25 21:59:30 +00:00
|
|
|
|
|
|
|
return (
|
2023-12-19 12:22:15 +00:00
|
|
|
<div className="flex flex-col gap-3 pt-4" data-testid="current-volume">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardStat
|
2023-12-21 10:38:09 +00:00
|
|
|
value={formatNumberRounded(currentVolume)}
|
2023-12-05 20:01:32 +00:00
|
|
|
text={t('pastEpochs', 'Past {{count}} epochs', {
|
|
|
|
count: windowLength,
|
|
|
|
})}
|
2023-12-19 12:22:15 +00:00
|
|
|
testId="past-epochs-volume"
|
2023-10-25 21:59:30 +00:00
|
|
|
/>
|
2023-12-05 19:36:25 +00:00
|
|
|
{requiredForNextTier.isGreaterThan(0) && (
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardStat
|
2023-10-25 21:59:30 +00:00
|
|
|
value={formatNumber(requiredForNextTier)}
|
|
|
|
text={t('Required for next tier')}
|
2023-12-19 12:22:15 +00:00
|
|
|
testId="required-for-next-tier"
|
2023-10-25 21:59:30 +00:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const ReferralBenefits = ({
|
|
|
|
epochsInSet,
|
|
|
|
setRunningNotionalTakerVolume,
|
|
|
|
epochs,
|
|
|
|
}: {
|
|
|
|
epochsInSet: number;
|
|
|
|
setRunningNotionalTakerVolume: number;
|
|
|
|
epochs: number;
|
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-10-25 21:59:30 +00:00
|
|
|
return (
|
2023-12-19 12:22:15 +00:00
|
|
|
<div className="flex flex-col gap-3 pt-4" data-testid="referral-benefits">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardStat
|
2023-10-25 21:59:30 +00:00
|
|
|
// all sets volume (not just current party)
|
|
|
|
value={formatNumber(setRunningNotionalTakerVolume)}
|
2023-11-30 07:31:44 +00:00
|
|
|
text={t(
|
|
|
|
'runningNotionalOverEpochs',
|
|
|
|
'Combined running notional over the {{count}} epochs',
|
|
|
|
{
|
|
|
|
count: epochs,
|
|
|
|
}
|
|
|
|
)}
|
2023-12-19 12:22:15 +00:00
|
|
|
testId="running-notional-taker-volume"
|
|
|
|
/>
|
|
|
|
<CardStat
|
|
|
|
value={epochsInSet}
|
|
|
|
text={t('epochs in referral set')}
|
|
|
|
testId="epochs-in-referral-set"
|
2023-10-25 21:59:30 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const TotalDiscount = ({
|
|
|
|
referralDiscount,
|
|
|
|
volumeDiscount,
|
2023-11-09 15:50:50 +00:00
|
|
|
isReferralProgramRunning,
|
|
|
|
isVolumeDiscountProgramRunning,
|
2023-10-25 21:59:30 +00:00
|
|
|
}: {
|
|
|
|
referralDiscount: number;
|
|
|
|
volumeDiscount: number;
|
2023-11-09 15:50:50 +00:00
|
|
|
isReferralProgramRunning: boolean;
|
|
|
|
isVolumeDiscountProgramRunning: boolean;
|
2023-10-25 21:59:30 +00:00
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-11-09 15:50:50 +00:00
|
|
|
const totalDiscount = 1 - (1 - volumeDiscount) * (1 - referralDiscount);
|
|
|
|
const totalDiscountDescription = t(
|
|
|
|
'The total discount is calculated according to the following formula: '
|
|
|
|
);
|
|
|
|
const formula = (
|
|
|
|
<span className="italic">
|
|
|
|
1 - (1 - d<sub>volume</sub>) ⋇ (1 - d<sub>referral</sub>)
|
|
|
|
</span>
|
|
|
|
);
|
|
|
|
|
2023-10-25 21:59:30 +00:00
|
|
|
return (
|
2023-12-19 12:22:15 +00:00
|
|
|
<div className="pt-4" data-testid="total-discount-card-stats">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardStat
|
2023-11-09 15:50:50 +00:00
|
|
|
description={
|
|
|
|
<>
|
|
|
|
{totalDiscountDescription}
|
|
|
|
{formula}
|
|
|
|
</>
|
|
|
|
}
|
|
|
|
value={formatPercentage(totalDiscount) + '%'}
|
2023-10-25 21:59:30 +00:00
|
|
|
highlight={true}
|
2023-12-19 12:22:15 +00:00
|
|
|
testId="total-discount"
|
2023-10-25 21:59:30 +00:00
|
|
|
/>
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTable>
|
2023-12-19 12:22:15 +00:00
|
|
|
<tr data-testid="volume-discount-row">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTableTH>{t('Volume discount')}</CardTableTH>
|
|
|
|
<CardTableTD>
|
|
|
|
{formatPercentage(volumeDiscount)}%
|
|
|
|
{!isVolumeDiscountProgramRunning && (
|
|
|
|
<Tooltip description={t('No active volume discount programme')}>
|
|
|
|
<span className="cursor-help">
|
|
|
|
{' '}
|
|
|
|
<VegaIcon name={VegaIconNames.INFO} size={12} />
|
|
|
|
</span>
|
|
|
|
</Tooltip>
|
|
|
|
)}
|
|
|
|
</CardTableTD>
|
|
|
|
</tr>
|
2023-12-19 12:22:15 +00:00
|
|
|
<tr data-testid="referral-discount-row">
|
2023-11-15 21:46:19 +00:00
|
|
|
<CardTableTH>{t('Referral discount')}</CardTableTH>
|
|
|
|
<CardTableTD>
|
|
|
|
{formatPercentage(referralDiscount)}%
|
|
|
|
{!isReferralProgramRunning && (
|
|
|
|
<Tooltip description={t('No active referral programme')}>
|
|
|
|
<span className="cursor-help">
|
|
|
|
{' '}
|
|
|
|
<VegaIcon name={VegaIconNames.INFO} size={12} />
|
|
|
|
</span>
|
|
|
|
</Tooltip>
|
|
|
|
)}
|
|
|
|
</CardTableTD>
|
|
|
|
</tr>
|
|
|
|
</CardTable>
|
2023-10-25 21:59:30 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const VolumeTiers = ({
|
|
|
|
tiers,
|
|
|
|
tierIndex,
|
|
|
|
lastEpochVolume,
|
2023-11-03 15:11:51 +00:00
|
|
|
windowLength,
|
2023-10-25 21:59:30 +00:00
|
|
|
}: {
|
|
|
|
tiers: Array<{
|
|
|
|
volumeDiscountFactor: string;
|
|
|
|
minimumRunningNotionalTakerVolume: string;
|
|
|
|
}>;
|
|
|
|
tierIndex: number;
|
|
|
|
lastEpochVolume: number;
|
2023-11-03 15:11:51 +00:00
|
|
|
windowLength: number;
|
2023-10-25 21:59:30 +00:00
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-10-25 21:59:30 +00:00
|
|
|
if (!tiers.length) {
|
|
|
|
return (
|
2023-11-15 21:46:19 +00:00
|
|
|
<p className="text-muted text-sm">
|
2023-10-25 21:59:30 +00:00
|
|
|
{t('No volume discount program active')}
|
|
|
|
</p>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div>
|
2024-01-02 15:50:04 +00:00
|
|
|
<SimpleTable
|
|
|
|
className="bg-white dark:bg-vega-cdark-900"
|
|
|
|
columns={[
|
|
|
|
{ name: 'tier', displayName: t('Tier'), testId: 'col-tier-value' },
|
|
|
|
{
|
|
|
|
name: 'discount',
|
|
|
|
displayName: t('Discount'),
|
|
|
|
testId: 'discount-value',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'minTradingVolume',
|
|
|
|
displayName: t('Min. trading volume'),
|
|
|
|
testId: 'min-volume-value',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'myVolume',
|
|
|
|
displayName: t('myVolume', 'My volume (last {{count}} epochs)', {
|
|
|
|
count: windowLength,
|
|
|
|
}),
|
|
|
|
testId: 'my-volume-value',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'indicator',
|
|
|
|
className: 'max-md:hidden',
|
|
|
|
testId: 'your-tier',
|
|
|
|
},
|
|
|
|
]}
|
|
|
|
data={Array.from(tiers).map((tier, i) => {
|
|
|
|
const isUserTier = tierIndex === i;
|
|
|
|
const indicator = isUserTier ? (
|
|
|
|
<YourTier testId={`your-volume-tier-${i}`} />
|
|
|
|
) : null;
|
|
|
|
const tierIndicator = (
|
|
|
|
<div className="flex justify-between">
|
|
|
|
<span data-testid={`tier-value-${i}`}>{i + 1}</span>
|
|
|
|
<span className="md:hidden">{indicator}</span>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
return {
|
|
|
|
tier: tierIndicator,
|
|
|
|
discount: (
|
|
|
|
<>{formatPercentage(Number(tier.volumeDiscountFactor))}%</>
|
|
|
|
),
|
|
|
|
minTradingVolume: (
|
|
|
|
<>{formatNumber(tier.minimumRunningNotionalTakerVolume)}</>
|
|
|
|
),
|
|
|
|
myVolume: isUserTier ? (
|
|
|
|
formatNumber(lastEpochVolume)
|
|
|
|
) : (
|
|
|
|
<span className="md:hidden">-</span>
|
|
|
|
),
|
|
|
|
indicator: indicator,
|
|
|
|
className: classNames(
|
|
|
|
getTierGradient(i + 1, tiers.length),
|
|
|
|
'text-xs'
|
|
|
|
),
|
|
|
|
};
|
|
|
|
})}
|
|
|
|
/>
|
2023-10-25 21:59:30 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const ReferralTiers = ({
|
|
|
|
tiers,
|
|
|
|
tierIndex,
|
|
|
|
epochsInSet,
|
|
|
|
referralVolumeInWindow,
|
2024-01-02 15:50:04 +00:00
|
|
|
referralDiscountWindowLength,
|
2023-10-25 21:59:30 +00:00
|
|
|
}: {
|
|
|
|
tiers: Array<{
|
|
|
|
referralDiscountFactor: string;
|
|
|
|
minimumRunningNotionalTakerVolume: string;
|
|
|
|
minimumEpochs: number;
|
|
|
|
}>;
|
|
|
|
tierIndex: number;
|
|
|
|
epochsInSet: number;
|
|
|
|
referralVolumeInWindow: number;
|
2024-01-02 15:50:04 +00:00
|
|
|
referralDiscountWindowLength: number;
|
2023-10-25 21:59:30 +00:00
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
|
|
|
|
2023-10-25 21:59:30 +00:00
|
|
|
if (!tiers.length) {
|
|
|
|
return (
|
2023-11-15 21:46:19 +00:00
|
|
|
<p className="text-muted text-sm">{t('No referral program active')}</p>
|
2023-10-25 21:59:30 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div>
|
2024-01-02 15:50:04 +00:00
|
|
|
<SimpleTable
|
|
|
|
className="bg-white dark:bg-vega-cdark-900"
|
|
|
|
columns={[
|
|
|
|
{ name: 'tier', displayName: t('Tier'), testId: 'col-tier-value' },
|
|
|
|
{
|
|
|
|
name: 'discount',
|
|
|
|
displayName: t('Discount'),
|
|
|
|
tooltip: t(
|
|
|
|
"The proportion of the referee's taker fees to be discounted"
|
|
|
|
),
|
|
|
|
testId: 'discount-value',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'volume',
|
|
|
|
displayName: t(
|
|
|
|
'minTradingVolume',
|
|
|
|
'Min. trading volume (last {{count}} epochs)',
|
|
|
|
{
|
|
|
|
count: referralDiscountWindowLength,
|
|
|
|
}
|
|
|
|
),
|
|
|
|
tooltip: t(
|
|
|
|
'The minimum running notional for the given benefit tier'
|
|
|
|
),
|
|
|
|
testId: 'min-volume-value',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'epochs',
|
|
|
|
displayName: t('Min. epochs'),
|
|
|
|
tooltip: t(
|
|
|
|
'The minimum number of epochs the party needs to be in the referral set to be eligible for the benefit'
|
|
|
|
),
|
|
|
|
testId: 'required-epochs-value',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'indicator',
|
|
|
|
className: 'max-md:hidden',
|
|
|
|
testId: 'user-tier-or-unlocks',
|
|
|
|
},
|
|
|
|
]}
|
|
|
|
data={Array.from(tiers).map((tier, i) => {
|
|
|
|
const isUserTier = tierIndex === i;
|
|
|
|
const requiredVolume = Number(tier.minimumRunningNotionalTakerVolume);
|
|
|
|
|
|
|
|
const indicator = isUserTier ? (
|
|
|
|
<YourTier testId={`your-referral-tier-${i}`} />
|
|
|
|
) : referralVolumeInWindow >= requiredVolume &&
|
|
|
|
epochsInSet < tier.minimumEpochs ? (
|
|
|
|
<span className="text-muted text-xs">
|
|
|
|
Unlocks in {tier.minimumEpochs - epochsInSet} epochs
|
|
|
|
</span>
|
|
|
|
) : null;
|
|
|
|
|
|
|
|
const tierIndicator = (
|
|
|
|
<div className="flex justify-between">
|
|
|
|
<span data-testid={`tier-value-${i}`}>{i + 1}</span>
|
|
|
|
<span className="md:hidden">{indicator}</span>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
|
|
|
return {
|
|
|
|
tier: tierIndicator,
|
|
|
|
discount: (
|
|
|
|
<>{formatPercentage(Number(tier.referralDiscountFactor))}%</>
|
|
|
|
),
|
|
|
|
volume: formatNumber(tier.minimumRunningNotionalTakerVolume),
|
|
|
|
epochs: tier.minimumEpochs,
|
|
|
|
indicator,
|
|
|
|
className: classNames(
|
|
|
|
getTierGradient(i + 1, tiers.length),
|
|
|
|
'text-xs'
|
|
|
|
),
|
|
|
|
};
|
|
|
|
})}
|
|
|
|
/>
|
2023-10-25 21:59:30 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-12-19 12:22:15 +00:00
|
|
|
interface YourTierProps {
|
|
|
|
testId?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
const YourTier = ({ testId }: YourTierProps) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
|
|
|
|
2023-10-25 21:59:30 +00:00
|
|
|
return (
|
2023-12-19 12:22:15 +00:00
|
|
|
<span
|
2024-01-02 15:50:04 +00:00
|
|
|
className="bg-rainbow whitespace-nowrap rounded-xl px-4 py-1.5 text-white text-xs"
|
2023-12-19 12:22:15 +00:00
|
|
|
data-testid={testId}
|
|
|
|
>
|
2023-10-25 21:59:30 +00:00
|
|
|
{t('Your tier')}
|
|
|
|
</span>
|
|
|
|
);
|
|
|
|
};
|
2023-11-09 15:50:50 +00:00
|
|
|
|
2023-11-16 03:10:39 +00:00
|
|
|
const ReferrerInfo = ({ code }: { code?: string }) => {
|
|
|
|
const t = useT();
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="text-vega-clight-200 dark:vega-cdark-200 pt-3 text-sm">
|
|
|
|
<p className="mb-1">
|
|
|
|
{t('Connected key is owner of the referral set')}
|
|
|
|
{code && (
|
|
|
|
<>
|
|
|
|
{' '}
|
|
|
|
<span className="bg-rainbow bg-clip-text text-transparent">
|
|
|
|
{truncateMiddle(code)}
|
|
|
|
</span>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
{'. '}
|
|
|
|
{t('As owner, it is eligible for commission not fee discounts.')}
|
|
|
|
</p>
|
|
|
|
<p>
|
|
|
|
{t('See')}{' '}
|
|
|
|
<Link
|
|
|
|
className="text-black underline dark:text-white"
|
|
|
|
to={Links.REFERRALS()}
|
|
|
|
>
|
|
|
|
{t('Referrals')}
|
|
|
|
</Link>{' '}
|
|
|
|
{t('for more information.')}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|