feat(accounts): accounts table quantum formatting (#4076)
Co-authored-by: Bartłomiej Głownia <bglownia@gmail.com>
This commit is contained in:
parent
c8d55c6293
commit
98d248e02e
@ -1,5 +1,5 @@
|
|||||||
// #region consts
|
// #region consts
|
||||||
const asset = 'asset';
|
const assetColId = '[col-id="asset.symbol"]';
|
||||||
const assetDetailsDialog = 'dialog-content';
|
const assetDetailsDialog = 'dialog-content';
|
||||||
const assetRow = 'key-value-table-row';
|
const assetRow = 'key-value-table-row';
|
||||||
const contractAddress = '7_value';
|
const contractAddress = '7_value';
|
||||||
@ -108,7 +108,7 @@ beforeEach(() => {
|
|||||||
|
|
||||||
const visitPortfolioAndClickAsset = (assetName: string) => {
|
const visitPortfolioAndClickAsset = (assetName: string) => {
|
||||||
cy.visit('/#/portfolio');
|
cy.visit('/#/portfolio');
|
||||||
cy.getByTestId(asset).contains(assetName).click();
|
cy.get(assetColId).contains(assetName).click();
|
||||||
};
|
};
|
||||||
|
|
||||||
const testTooltip = (index: number, testId: string, tooltip: string) => {
|
const testTooltip = (index: number, testId: string, tooltip: string) => {
|
||||||
|
@ -20,6 +20,7 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
// 7001-COLL-006
|
// 7001-COLL-006
|
||||||
// 7001-COLL-007
|
// 7001-COLL-007
|
||||||
// 1003-TRAN-001
|
// 1003-TRAN-001
|
||||||
|
// 7001-COLL-012
|
||||||
|
|
||||||
const tradingAccountRowId = '[row-id="t-0"]';
|
const tradingAccountRowId = '[row-id="t-0"]';
|
||||||
cy.getByTestId('Collateral').click();
|
cy.getByTestId('Collateral').click();
|
||||||
@ -34,28 +35,23 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
cy.getByTestId('tab-accounts')
|
cy.getByTestId('tab-accounts')
|
||||||
.get(tradingAccountRowId)
|
.get(tradingAccountRowId)
|
||||||
.find('[col-id="used"]')
|
.find('[col-id="used"]')
|
||||||
.should('have.text', '1.010.00%');
|
.should('have.text', '1.01' + '1.00%');
|
||||||
|
|
||||||
cy.getByTestId('tab-accounts')
|
cy.getByTestId('tab-accounts')
|
||||||
.get(tradingAccountRowId)
|
.get(tradingAccountRowId)
|
||||||
.find('[col-id="available"]')
|
.find('[col-id="available"]')
|
||||||
.should('have.text', '100,000.00');
|
.should('have.text', '100.00');
|
||||||
|
|
||||||
cy.getByTestId('tab-accounts')
|
cy.getByTestId('tab-accounts')
|
||||||
.get(tradingAccountRowId)
|
.get(tradingAccountRowId)
|
||||||
.find('[col-id="total"]')
|
.find('[col-id="total"]')
|
||||||
.should('have.text', '100,001.01');
|
.should('have.text', '101.01');
|
||||||
|
|
||||||
cy.getByTestId('tab-accounts')
|
cy.getByTestId('tab-accounts')
|
||||||
.get(tradingAccountRowId)
|
.get(tradingAccountRowId)
|
||||||
.find('[col-id="accounts-actions"]')
|
.find('[col-id="accounts-actions"]')
|
||||||
.should('have.text', '');
|
.should('have.text', '');
|
||||||
|
|
||||||
cy.getByTestId('tab-accounts')
|
|
||||||
.get(tradingAccountRowId)
|
|
||||||
.find('[col-id="total"]')
|
|
||||||
.should('have.text', '100,001.01');
|
|
||||||
|
|
||||||
cy.getByTestId('tab-accounts')
|
cy.getByTestId('tab-accounts')
|
||||||
.get('[col-id="accounts-actions"]')
|
.get('[col-id="accounts-actions"]')
|
||||||
.find('[data-testid="dropdown-menu"]')
|
.find('[data-testid="dropdown-menu"]')
|
||||||
@ -101,7 +97,7 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
'Liquidity provision fee reward account balance',
|
'Liquidity provision fee reward account balance',
|
||||||
'Market proposer reward account balance',
|
'Market proposer reward account balance',
|
||||||
];
|
];
|
||||||
cy.getByTestId('asset').contains('tEURO').click();
|
cy.get('[col-id="asset.symbol"]').contains('tEURO').click();
|
||||||
cy.get('[data-testid$="_label"]').should('have.length', 16);
|
cy.get('[data-testid$="_label"]').should('have.length', 16);
|
||||||
cy.get('[data-testid$="_label"]').each((element, index) => {
|
cy.get('[data-testid$="_label"]').each((element, index) => {
|
||||||
cy.wrap(element).should('have.text', titles[index]);
|
cy.wrap(element).should('have.text', titles[index]);
|
||||||
@ -112,7 +108,7 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
|
|
||||||
it('should open usage breakdown dialog when clicked on used', () => {
|
it('should open usage breakdown dialog when clicked on used', () => {
|
||||||
// 7001-COLL-009
|
// 7001-COLL-009
|
||||||
cy.getByTestId('breakdown').contains('1.01').click();
|
cy.get('[col-id="used"]').contains('1.01').click();
|
||||||
const headers = ['Market', 'Account type', 'Balance'];
|
const headers = ['Market', 'Account type', 'Balance'];
|
||||||
cy.getByTestId('usage-breakdown').within(($headers) => {
|
cy.getByTestId('usage-breakdown').within(($headers) => {
|
||||||
cy.wrap($headers)
|
cy.wrap($headers)
|
||||||
@ -137,7 +133,7 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
cy.getByTestId('Collateral').click();
|
cy.getByTestId('Collateral').click();
|
||||||
const marketsSortedDefault = ['tBTC', 'tEURO', 'tDAI', 'tBTC'];
|
const marketsSortedDefault = ['tBTC', 'tEURO', 'tDAI', 'tBTC'];
|
||||||
const marketsSortedAsc = ['tBTC', 'tBTC', 'tDAI', 'tEURO'];
|
const marketsSortedAsc = ['tBTC', 'tBTC', 'tDAI', 'tEURO'];
|
||||||
const marketsSortedDesc = ['tEURO', 'tDAI', 'tBTC', 'tBTC'];
|
const marketsSortedDesc = Array.from(marketsSortedAsc).reverse();
|
||||||
checkSorting(
|
checkSorting(
|
||||||
'asset.symbol',
|
'asset.symbol',
|
||||||
marketsSortedDefault,
|
marketsSortedDefault,
|
||||||
@ -149,23 +145,14 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
it('sorting by total', () => {
|
it('sorting by total', () => {
|
||||||
cy.getByTestId('Collateral').click();
|
cy.getByTestId('Collateral').click();
|
||||||
const marketsSortedDefault = [
|
const marketsSortedDefault = [
|
||||||
'1,000.00002',
|
'1,000.00',
|
||||||
'1,000.01',
|
'1,000.01',
|
||||||
'1,000.00',
|
'1,000.00',
|
||||||
'1,000.00001',
|
|
||||||
];
|
|
||||||
const marketsSortedAsc = [
|
|
||||||
'1,000.00',
|
|
||||||
'1,000.00001',
|
|
||||||
'1,000.00002',
|
|
||||||
'1,000.01',
|
|
||||||
];
|
|
||||||
const marketsSortedDesc = [
|
|
||||||
'1,000.01',
|
|
||||||
'1,000.00002',
|
|
||||||
'1,000.00001',
|
|
||||||
'1,000.00',
|
'1,000.00',
|
||||||
];
|
];
|
||||||
|
const marketsSortedAsc = ['1,000.00', '1,000.00', '1,000.00', '1,000.01'];
|
||||||
|
const marketsSortedDesc = Array.from(marketsSortedAsc).reverse();
|
||||||
|
|
||||||
checkSorting(
|
checkSorting(
|
||||||
'total',
|
'total',
|
||||||
marketsSortedDefault,
|
marketsSortedDefault,
|
||||||
@ -176,24 +163,22 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
|
|
||||||
it('sorting by used', () => {
|
it('sorting by used', () => {
|
||||||
cy.getByTestId('Collateral').click();
|
cy.getByTestId('Collateral').click();
|
||||||
|
// concat actual value with percentage value
|
||||||
|
// as cypress will pick up the entire cell contes
|
||||||
|
// textContent
|
||||||
const marketsSortedDefault = [
|
const marketsSortedDefault = [
|
||||||
'0.000.00%',
|
'0.00' + '0.00%',
|
||||||
'0.010.00%',
|
'0.01' + '0.00%',
|
||||||
'0.000.00%',
|
'0.00' + '0.00%',
|
||||||
'0.000.00%',
|
'0.00' + '0.00%',
|
||||||
];
|
];
|
||||||
const marketsSortedAsc = [
|
const marketsSortedAsc = [
|
||||||
'0.000.00%',
|
'0.00' + '0.00%',
|
||||||
'0.000.00%',
|
'0.00' + '0.00%',
|
||||||
'0.000.00%',
|
'0.00' + '0.00%',
|
||||||
'0.010.00%',
|
'0.01' + '0.00%',
|
||||||
];
|
|
||||||
const marketsSortedDesc = [
|
|
||||||
'0.010.00%',
|
|
||||||
'0.000.00%',
|
|
||||||
'0.000.00%',
|
|
||||||
'0.000.00%',
|
|
||||||
];
|
];
|
||||||
|
const marketsSortedDesc = Array.from(marketsSortedAsc).reverse();
|
||||||
checkSorting(
|
checkSorting(
|
||||||
'used',
|
'used',
|
||||||
marketsSortedDefault,
|
marketsSortedDefault,
|
||||||
@ -205,23 +190,13 @@ describe('accounts', { tags: '@smoke' }, () => {
|
|||||||
it('sorting by total', () => {
|
it('sorting by total', () => {
|
||||||
cy.getByTestId('Collateral').click();
|
cy.getByTestId('Collateral').click();
|
||||||
const marketsSortedDefault = [
|
const marketsSortedDefault = [
|
||||||
'1,000.00002',
|
'1,000.00',
|
||||||
'1,000.01',
|
'1,000.01',
|
||||||
'1,000.00',
|
'1,000.00',
|
||||||
'1,000.00001',
|
|
||||||
];
|
|
||||||
const marketsSortedAsc = [
|
|
||||||
'1,000.00',
|
|
||||||
'1,000.00001',
|
|
||||||
'1,000.00002',
|
|
||||||
'1,000.01',
|
|
||||||
];
|
|
||||||
const marketsSortedDesc = [
|
|
||||||
'1,000.01',
|
|
||||||
'1,000.00002',
|
|
||||||
'1,000.00001',
|
|
||||||
'1,000.00',
|
'1,000.00',
|
||||||
];
|
];
|
||||||
|
const marketsSortedAsc = ['1,000.00', '1,000.00', '1,000.00', '1,000.01'];
|
||||||
|
const marketsSortedDesc = Array.from(marketsSortedAsc).reverse();
|
||||||
|
|
||||||
checkSorting(
|
checkSorting(
|
||||||
'total',
|
'total',
|
||||||
|
@ -102,7 +102,7 @@ describe('deal ticker order validation', { tags: '@smoke' }, () => {
|
|||||||
.within(() => {
|
.within(() => {
|
||||||
cy.get('[data-state="closed"]').should(
|
cy.get('[data-state="closed"]').should(
|
||||||
'have.text',
|
'have.text',
|
||||||
'Total margin available100,000.01 tDAI'
|
'Total margin available' + '100.01 tDAI'
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -293,43 +293,36 @@ describe('positions', { tags: '@regression', testIsolation: true }, () => {
|
|||||||
});
|
});
|
||||||
function validatePositionsDisplayed(multiKey = false) {
|
function validatePositionsDisplayed(multiKey = false) {
|
||||||
cy.getByTestId('tab-positions').should('be.visible');
|
cy.getByTestId('tab-positions').should('be.visible');
|
||||||
cy.getByTestId('tab-positions').within(() => {
|
cy.getByTestId('tab-positions')
|
||||||
cy.get('[col-id="marketName"]')
|
.get('.ag-center-cols-container .ag-row')
|
||||||
.should('be.visible')
|
.first()
|
||||||
.each(($marketSymbol) => {
|
.within(() => {
|
||||||
cy.wrap($marketSymbol).invoke('text').should('not.be.empty');
|
cy.get('[col-id="marketName"]')
|
||||||
});
|
.should('be.visible')
|
||||||
|
.invoke('text')
|
||||||
|
.should('not.be.empty');
|
||||||
|
|
||||||
cy.get('.ag-center-cols-container [col-id="openVolume"]').each(
|
cy.get('[col-id="openVolume"]').should('not.be.empty');
|
||||||
($openVolume) => {
|
|
||||||
cy.wrap($openVolume).invoke('text').should('not.be.empty');
|
// includes average entry price, mark price, realised PNL & leverage
|
||||||
|
cy.getByTestId('flash-cell').should('not.be.empty');
|
||||||
|
|
||||||
|
if (!multiKey) {
|
||||||
|
cy.get('[col-id="currentLeverage"]').should('contain.text', '2,767.3');
|
||||||
|
cy.get('[col-id="marginAccountBalance"]') // margin allocated
|
||||||
|
.should('contain.text', '0.01');
|
||||||
}
|
}
|
||||||
);
|
|
||||||
|
|
||||||
// includes average entry price, mark price, realised PNL & leverage
|
cy.get('[col-id="unrealisedPNL"]').should('not.be.empty');
|
||||||
cy.getByTestId('flash-cell').each(($prices) => {
|
cy.get('[col-id="notional"]').should('contain.text', '276,761.40348'); // Total tDAI position
|
||||||
cy.wrap($prices).invoke('text').should('not.be.empty');
|
cy.get('[col-id="realisedPNL"]').should('contain.text', '2.30'); // Total Realised PNL
|
||||||
|
cy.get('[col-id="unrealisedPNL"]').should('contain.text', '8.95'); // Total Unrealised PNL
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!multiKey) {
|
cy.get('.ag-header-row [col-id="notional"]')
|
||||||
cy.get('[col-id="currentLeverage"]').should('contain.text', '2.846.1');
|
.should('contain.text', 'Notional')
|
||||||
cy.get('[col-id="marginAccountBalance"]') // margin allocated
|
.realHover();
|
||||||
.should('contain.text', '0.01');
|
cy.get('.ag-popup').should('contain.text', 'Mark price x open volume');
|
||||||
}
|
|
||||||
|
|
||||||
cy.get('[col-id="unrealisedPNL"]').each(($unrealisedPnl) => {
|
|
||||||
cy.wrap($unrealisedPnl).invoke('text').should('not.be.empty');
|
|
||||||
});
|
|
||||||
|
|
||||||
cy.get('[col-id="notional"]').should('contain.text', '276,761.40348'); // Total tDAI position
|
|
||||||
cy.get('[col-id="realisedPNL"]').should('contain.text', '2.30'); // Total Realised PNL
|
|
||||||
cy.get('[col-id="unrealisedPNL"]').should('contain.text', '8.95'); // Total Unrealised PNL
|
|
||||||
|
|
||||||
cy.get('.ag-header-row [col-id="notional"]')
|
|
||||||
.should('contain.text', 'Notional')
|
|
||||||
.realHover();
|
|
||||||
cy.get('.ag-popup').should('contain.text', 'Mark price x open volume');
|
|
||||||
});
|
|
||||||
|
|
||||||
cy.getByTestId('close-position').should('be.visible').and('have.length', 3);
|
cy.getByTestId('close-position').should('be.visible').and('have.length', 3);
|
||||||
}
|
}
|
||||||
|
@ -143,7 +143,6 @@ const MarketBottomPanel = memo(
|
|||||||
<VegaWalletContainer>
|
<VegaWalletContainer>
|
||||||
<TradingViews.collateral.component
|
<TradingViews.collateral.component
|
||||||
pinnedAsset={pinnedAsset}
|
pinnedAsset={pinnedAsset}
|
||||||
noBottomPlaceholder
|
|
||||||
hideButtons
|
hideButtons
|
||||||
storeKey="marketCollateral"
|
storeKey="marketCollateral"
|
||||||
/>
|
/>
|
||||||
|
@ -12,12 +12,10 @@ import { useDepositDialog } from '@vegaprotocol/deposits';
|
|||||||
export const AccountsContainer = ({
|
export const AccountsContainer = ({
|
||||||
pinnedAsset,
|
pinnedAsset,
|
||||||
hideButtons,
|
hideButtons,
|
||||||
noBottomPlaceholder,
|
|
||||||
storeKey,
|
storeKey,
|
||||||
}: {
|
}: {
|
||||||
pinnedAsset?: PinnedAsset;
|
pinnedAsset?: PinnedAsset;
|
||||||
hideButtons?: boolean;
|
hideButtons?: boolean;
|
||||||
noBottomPlaceholder?: boolean;
|
|
||||||
storeKey?: string;
|
storeKey?: string;
|
||||||
}) => {
|
}) => {
|
||||||
const { pubKey, isReadOnly } = useVegaWallet();
|
const { pubKey, isReadOnly } = useVegaWallet();
|
||||||
@ -50,7 +48,6 @@ export const AccountsContainer = ({
|
|||||||
onClickDeposit={openDepositDialog}
|
onClickDeposit={openDepositDialog}
|
||||||
isReadOnly={isReadOnly}
|
isReadOnly={isReadOnly}
|
||||||
pinnedAsset={pinnedAsset}
|
pinnedAsset={pinnedAsset}
|
||||||
noBottomPlaceholder={noBottomPlaceholder}
|
|
||||||
storeKey={storeKey}
|
storeKey={storeKey}
|
||||||
/>
|
/>
|
||||||
{!isReadOnly && !hideButtons && (
|
{!isReadOnly && !hideButtons && (
|
||||||
|
@ -1,17 +1,14 @@
|
|||||||
import { useRef, memo, useCallback, useState } from 'react';
|
import { useRef, memo, useState } from 'react';
|
||||||
import { addDecimalsFormatNumber } from '@vegaprotocol/utils';
|
import { addDecimalsFormatNumber } from '@vegaprotocol/utils';
|
||||||
import { t } from '@vegaprotocol/i18n';
|
import { t } from '@vegaprotocol/i18n';
|
||||||
import { useBottomPlaceholder } from '@vegaprotocol/datagrid';
|
|
||||||
import { useDataProvider } from '@vegaprotocol/data-provider';
|
import { useDataProvider } from '@vegaprotocol/data-provider';
|
||||||
import type { AgGridReact } from 'ag-grid-react';
|
import type { AgGridReact } from 'ag-grid-react';
|
||||||
import type { AccountFields } from './accounts-data-provider';
|
|
||||||
import {
|
import {
|
||||||
aggregatedAccountsDataProvider,
|
aggregatedAccountsDataProvider,
|
||||||
aggregatedAccountDataProvider,
|
aggregatedAccountDataProvider,
|
||||||
} from './accounts-data-provider';
|
} from './accounts-data-provider';
|
||||||
import type { PinnedAsset } from './accounts-table';
|
import type { PinnedAsset } from './accounts-table';
|
||||||
import { AccountTable } from './accounts-table';
|
import { AccountTable } from './accounts-table';
|
||||||
import isEqual from 'lodash/isEqual';
|
|
||||||
import { Dialog } from '@vegaprotocol/ui-toolkit';
|
import { Dialog } from '@vegaprotocol/ui-toolkit';
|
||||||
import BreakdownTable from './breakdown-table';
|
import BreakdownTable from './breakdown-table';
|
||||||
|
|
||||||
@ -54,7 +51,6 @@ interface AccountManagerProps {
|
|||||||
onClickDeposit?: (assetId?: string) => void;
|
onClickDeposit?: (assetId?: string) => void;
|
||||||
isReadOnly: boolean;
|
isReadOnly: boolean;
|
||||||
pinnedAsset?: PinnedAsset;
|
pinnedAsset?: PinnedAsset;
|
||||||
noBottomPlaceholder?: boolean;
|
|
||||||
storeKey?: string;
|
storeKey?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -65,48 +61,14 @@ export const AccountManager = ({
|
|||||||
partyId,
|
partyId,
|
||||||
isReadOnly,
|
isReadOnly,
|
||||||
pinnedAsset,
|
pinnedAsset,
|
||||||
noBottomPlaceholder,
|
|
||||||
storeKey,
|
storeKey,
|
||||||
}: AccountManagerProps) => {
|
}: AccountManagerProps) => {
|
||||||
const gridRef = useRef<AgGridReact | null>(null);
|
const gridRef = useRef<AgGridReact | null>(null);
|
||||||
const [breakdownAssetId, setBreakdownAssetId] = useState<string>();
|
const [breakdownAssetId, setBreakdownAssetId] = useState<string>();
|
||||||
const update = useCallback(
|
|
||||||
({ data }: { data: AccountFields[] | null }) => {
|
|
||||||
if (!data || !gridRef.current?.api) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
const pinnedAssetRowData =
|
|
||||||
pinnedAsset && data.find((d) => d.asset.id === pinnedAsset.id);
|
|
||||||
|
|
||||||
if (pinnedAssetRowData) {
|
|
||||||
const pinnedTopRow = gridRef.current.api.getPinnedTopRow(0);
|
|
||||||
if (
|
|
||||||
pinnedTopRow?.data?.balance === '0' &&
|
|
||||||
pinnedAssetRowData.balance !== '0'
|
|
||||||
) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (!isEqual(pinnedTopRow?.data, pinnedAssetRowData)) {
|
|
||||||
gridRef.current.api.setPinnedTopRowData([pinnedAssetRowData]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
gridRef.current.api.setRowData(
|
|
||||||
pinnedAssetRowData
|
|
||||||
? data?.filter((d) => d !== pinnedAssetRowData)
|
|
||||||
: data
|
|
||||||
);
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
[gridRef, pinnedAsset]
|
|
||||||
);
|
|
||||||
const { data, error } = useDataProvider({
|
const { data, error } = useDataProvider({
|
||||||
dataProvider: aggregatedAccountsDataProvider,
|
dataProvider: aggregatedAccountsDataProvider,
|
||||||
variables: { partyId },
|
variables: { partyId },
|
||||||
update,
|
|
||||||
});
|
|
||||||
const bottomPlaceholderProps = useBottomPlaceholder({
|
|
||||||
gridRef,
|
|
||||||
disabled: noBottomPlaceholder,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -121,7 +83,6 @@ export const AccountManager = ({
|
|||||||
isReadOnly={isReadOnly}
|
isReadOnly={isReadOnly}
|
||||||
pinnedAsset={pinnedAsset}
|
pinnedAsset={pinnedAsset}
|
||||||
storeKey={storeKey}
|
storeKey={storeKey}
|
||||||
{...bottomPlaceholderProps}
|
|
||||||
overlayNoRowsTemplate={error ? error.message : t('No accounts')}
|
overlayNoRowsTemplate={error ? error.message : t('No accounts')}
|
||||||
/>
|
/>
|
||||||
<Dialog
|
<Dialog
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import { forwardRef, useMemo, useCallback } from 'react';
|
import { forwardRef, useMemo, useCallback } from 'react';
|
||||||
import {
|
import {
|
||||||
addDecimalsFormatNumber,
|
addDecimalsFormatNumber,
|
||||||
|
addDecimalsFormatNumberQuantum,
|
||||||
isNumeric,
|
isNumeric,
|
||||||
toBigNum,
|
toBigNum,
|
||||||
} from '@vegaprotocol/utils';
|
} from '@vegaprotocol/utils';
|
||||||
@ -10,21 +11,15 @@ import type {
|
|||||||
VegaValueFormatterParams,
|
VegaValueFormatterParams,
|
||||||
} from '@vegaprotocol/datagrid';
|
} from '@vegaprotocol/datagrid';
|
||||||
import { COL_DEFS } from '@vegaprotocol/datagrid';
|
import { COL_DEFS } from '@vegaprotocol/datagrid';
|
||||||
import {
|
import { Button, VegaIcon, VegaIconNames } from '@vegaprotocol/ui-toolkit';
|
||||||
ButtonLink,
|
|
||||||
Button,
|
|
||||||
VegaIcon,
|
|
||||||
VegaIconNames,
|
|
||||||
} from '@vegaprotocol/ui-toolkit';
|
|
||||||
|
|
||||||
import { TooltipCellComponent } from '@vegaprotocol/ui-toolkit';
|
import { TooltipCellComponent } from '@vegaprotocol/ui-toolkit';
|
||||||
import { AgGridLazy as AgGrid } from '@vegaprotocol/datagrid';
|
import { AgGridLazy as AgGrid } from '@vegaprotocol/datagrid';
|
||||||
import { AgGridColumn } from 'ag-grid-react';
|
|
||||||
import type {
|
import type {
|
||||||
IDatasource,
|
|
||||||
IGetRowsParams,
|
IGetRowsParams,
|
||||||
RowNode,
|
RowNode,
|
||||||
RowHeightParams,
|
RowHeightParams,
|
||||||
|
ColDef,
|
||||||
} from 'ag-grid-community';
|
} from 'ag-grid-community';
|
||||||
import type { AgGridReact, AgGridReactProps } from 'ag-grid-react';
|
import type { AgGridReact, AgGridReactProps } from 'ag-grid-react';
|
||||||
import type { AccountFields } from './accounts-data-provider';
|
import type { AccountFields } from './accounts-data-provider';
|
||||||
@ -35,29 +30,16 @@ import classNames from 'classnames';
|
|||||||
import { AccountsActionsDropdown } from './accounts-actions-dropdown';
|
import { AccountsActionsDropdown } from './accounts-actions-dropdown';
|
||||||
|
|
||||||
const colorClass = (percentageUsed: number, neutral = false) => {
|
const colorClass = (percentageUsed: number, neutral = false) => {
|
||||||
return classNames({
|
return classNames('text-right', {
|
||||||
'text-neutral-500 dark:text-neutral-400': percentageUsed < 75 && !neutral,
|
'text-neutral-500 dark:text-neutral-400': percentageUsed < 75 && !neutral,
|
||||||
'text-vega-orange': percentageUsed >= 75 && percentageUsed < 90,
|
'text-vega-orange': percentageUsed >= 75 && percentageUsed < 90,
|
||||||
'text-vega-pink': percentageUsed >= 90,
|
'text-vega-pink': percentageUsed >= 90,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
export const percentageValue = (part?: string, total?: string) =>
|
export const percentageValue = (part: string, total: string) => {
|
||||||
new BigNumber(part || 0)
|
total = !total || total === '0' ? '1' : total;
|
||||||
.dividedBy(total || 1)
|
return new BigNumber(part).dividedBy(total).multipliedBy(100).toNumber();
|
||||||
.multipliedBy(100)
|
|
||||||
.toNumber();
|
|
||||||
|
|
||||||
const formatWithAssetDecimals = (
|
|
||||||
data: AccountFields | undefined,
|
|
||||||
value: string | undefined
|
|
||||||
) => {
|
|
||||||
return (
|
|
||||||
data &&
|
|
||||||
data.asset &&
|
|
||||||
isNumeric(value) &&
|
|
||||||
addDecimalsFormatNumber(value, data.asset.decimals)
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const accountValuesComparator = (
|
export const accountValuesComparator = (
|
||||||
@ -81,15 +63,10 @@ export interface GetRowsParams extends Omit<IGetRowsParams, 'successCallback'> {
|
|||||||
successCallback(rowsThisBlock: AccountFields[], lastRow?: number): void;
|
successCallback(rowsThisBlock: AccountFields[], lastRow?: number): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Datasource extends IDatasource {
|
|
||||||
getRows(params: GetRowsParams): void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type PinnedAsset = Pick<Asset, 'symbol' | 'name' | 'id' | 'decimals'>;
|
export type PinnedAsset = Pick<Asset, 'symbol' | 'name' | 'id' | 'decimals'>;
|
||||||
|
|
||||||
export interface AccountTableProps extends AgGridReactProps {
|
export interface AccountTableProps extends AgGridReactProps {
|
||||||
rowData?: AccountFields[] | null;
|
rowData?: AccountFields[] | null;
|
||||||
datasource?: Datasource;
|
|
||||||
onClickAsset: (assetId: string) => void;
|
onClickAsset: (assetId: string) => void;
|
||||||
onClickWithdraw?: (assetId: string) => void;
|
onClickWithdraw?: (assetId: string) => void;
|
||||||
onClickDeposit?: (assetId: string) => void;
|
onClickDeposit?: (assetId: string) => void;
|
||||||
@ -148,83 +125,55 @@ export const AccountTable = forwardRef<AgGridReact, AccountTableProps>(
|
|||||||
|
|
||||||
const showDepositButton = pinnedAsset?.balance === '0';
|
const showDepositButton = pinnedAsset?.balance === '0';
|
||||||
|
|
||||||
return (
|
const colDefs = useMemo(() => {
|
||||||
<AgGrid
|
const defs: ColDef[] = [
|
||||||
{...props}
|
{
|
||||||
style={{ width: '100%', height: '100%' }}
|
headerName: t('Asset'),
|
||||||
overlayNoRowsTemplate={t('No accounts')}
|
field: 'asset.symbol',
|
||||||
getRowId={({
|
headerTooltip: t(
|
||||||
data,
|
|
||||||
}: {
|
|
||||||
data: AccountFields & { isLastPlaceholder?: boolean; id?: string };
|
|
||||||
}) => (data.isLastPlaceholder && data.id ? data.id : data.asset.id)}
|
|
||||||
ref={ref}
|
|
||||||
tooltipShowDelay={500}
|
|
||||||
rowData={rowData?.filter(
|
|
||||||
(data) => data.asset.id !== props.pinnedAsset?.id
|
|
||||||
)}
|
|
||||||
defaultColDef={{
|
|
||||||
resizable: true,
|
|
||||||
tooltipComponent: TooltipCellComponent,
|
|
||||||
sortable: true,
|
|
||||||
comparator: accountValuesComparator,
|
|
||||||
}}
|
|
||||||
getRowHeight={getPinnedAssetRowHeight}
|
|
||||||
pinnedTopRowData={pinnedAsset ? [pinnedAsset] : undefined}
|
|
||||||
>
|
|
||||||
<AgGridColumn
|
|
||||||
headerName={t('Asset')}
|
|
||||||
field="asset.symbol"
|
|
||||||
headerTooltip={t(
|
|
||||||
'Asset is the collateral that is deposited into the Vega protocol.'
|
'Asset is the collateral that is deposited into the Vega protocol.'
|
||||||
)}
|
),
|
||||||
cellRenderer={({
|
cellClass: 'underline',
|
||||||
value,
|
onCellClicked: ({ data }) => {
|
||||||
data,
|
if (data) {
|
||||||
}: VegaICellRendererParams<AccountFields, 'asset.symbol'>) => {
|
onClickAsset(data.asset.id);
|
||||||
return (
|
}
|
||||||
<ButtonLink
|
},
|
||||||
data-testid="asset"
|
},
|
||||||
onClick={() => {
|
{
|
||||||
if (data) {
|
headerName: t('Used'),
|
||||||
onClickAsset(data.asset.id);
|
type: 'rightAligned',
|
||||||
}
|
field: 'used',
|
||||||
}}
|
headerTooltip: t(
|
||||||
>
|
|
||||||
{value}
|
|
||||||
</ButtonLink>
|
|
||||||
);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
<AgGridColumn
|
|
||||||
headerName={t('Used')}
|
|
||||||
type="rightAligned"
|
|
||||||
field="used"
|
|
||||||
headerTooltip={t(
|
|
||||||
'Currently allocated to a market as margin or bond. Check the breakdown for details.'
|
'Currently allocated to a market as margin or bond. Check the breakdown for details.'
|
||||||
)}
|
),
|
||||||
cellRenderer={({
|
tooltipValueGetter: ({ value, data }) => {
|
||||||
|
if (!value || !data) return null;
|
||||||
|
return addDecimalsFormatNumber(value, data.asset.decimals);
|
||||||
|
},
|
||||||
|
onCellClicked: ({ data }) => {
|
||||||
|
if (!data || !onClickBreakdown) return;
|
||||||
|
onClickBreakdown(data.asset.id);
|
||||||
|
},
|
||||||
|
cellRenderer: ({
|
||||||
data,
|
data,
|
||||||
value,
|
value,
|
||||||
}: VegaICellRendererParams<AccountFields, 'used'>) => {
|
}: VegaICellRendererParams<AccountFields, 'used'>) => {
|
||||||
if (!data) return null;
|
if (!value || !data) return '-';
|
||||||
const percentageUsed = percentageValue(value, data.total);
|
const percentageUsed = percentageValue(value, data.total);
|
||||||
const valueFormatted = formatWithAssetDecimals(data, value);
|
const valueFormatted = addDecimalsFormatNumberQuantum(
|
||||||
|
value,
|
||||||
|
data.asset.decimals,
|
||||||
|
data.asset.quantum
|
||||||
|
);
|
||||||
|
|
||||||
return data.breakdown ? (
|
return data.breakdown ? (
|
||||||
<>
|
<>
|
||||||
<ButtonLink
|
<span className="underline">{valueFormatted}</span>
|
||||||
data-testid="breakdown"
|
|
||||||
onClick={() => {
|
|
||||||
onClickBreakdown && onClickBreakdown(data.asset.id);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<span>{valueFormatted}</span>
|
|
||||||
</ButtonLink>
|
|
||||||
<span
|
<span
|
||||||
className={classNames(
|
className={classNames(
|
||||||
colorClass(percentageUsed),
|
colorClass(percentageUsed),
|
||||||
'ml-2 inline-block w-14'
|
'ml-1 inline-block w-14'
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
{percentageUsed.toFixed(2)}%
|
{percentageUsed.toFixed(2)}%
|
||||||
@ -232,59 +181,77 @@ export const AccountTable = forwardRef<AgGridReact, AccountTableProps>(
|
|||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
<span>{valueFormatted}</span>
|
<span className="underline">{valueFormatted}</span>
|
||||||
<span className="ml-2 inline-block w-14 text-neutral-500 dark:text-neutral-400">
|
<span className="ml-2 inline-block w-14 text-vega-light-200 dark:text-vega-dark-200">
|
||||||
0.00%
|
{t('0.00%')}'
|
||||||
</span>
|
</span>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}}
|
},
|
||||||
/>
|
},
|
||||||
<AgGridColumn
|
{
|
||||||
headerName={t('Available')}
|
headerName: t('Available'),
|
||||||
field="available"
|
field: 'available',
|
||||||
type="rightAligned"
|
type: 'rightAligned',
|
||||||
headerTooltip={t(
|
headerTooltip: t(
|
||||||
'Deposited on the network, but not allocated to a market. Free to use for placing orders or providing liquidity.'
|
'Deposited on the network, but not allocated to a market. Free to use for placing orders or providing liquidity.'
|
||||||
)}
|
),
|
||||||
cellRenderer={({
|
tooltipValueGetter: ({ value, data }) => {
|
||||||
|
if (!value || !data) return null;
|
||||||
|
return addDecimalsFormatNumber(value, data.asset.decimals);
|
||||||
|
},
|
||||||
|
cellClass: ({ data }) => {
|
||||||
|
const percentageUsed = percentageValue(data?.used, data?.total);
|
||||||
|
return colorClass(percentageUsed, true);
|
||||||
|
},
|
||||||
|
valueFormatter: ({
|
||||||
value,
|
value,
|
||||||
data,
|
data,
|
||||||
}: VegaICellRendererParams<AccountFields, 'available'>) => {
|
}: VegaValueFormatterParams<AccountFields, 'available'>) => {
|
||||||
const percentageUsed = percentageValue(data?.used, data?.total);
|
if (!value || !data) return '-';
|
||||||
|
return addDecimalsFormatNumberQuantum(
|
||||||
return (
|
value,
|
||||||
<span className={colorClass(percentageUsed, true)}>
|
data.asset.decimals,
|
||||||
{formatWithAssetDecimals(data, value)}
|
data.asset.quantum
|
||||||
</span>
|
|
||||||
);
|
);
|
||||||
}}
|
},
|
||||||
/>
|
},
|
||||||
<AgGridColumn
|
|
||||||
headerName={t('Total')}
|
{
|
||||||
type="rightAligned"
|
headerName: t('Total'),
|
||||||
field="total"
|
type: 'rightAligned',
|
||||||
headerTooltip={t(
|
field: 'total',
|
||||||
|
headerTooltip: t(
|
||||||
'The total amount of each asset on this key. Includes used and available collateral.'
|
'The total amount of each asset on this key. Includes used and available collateral.'
|
||||||
)}
|
),
|
||||||
valueFormatter={({
|
tooltipValueGetter: ({ value, data }) => {
|
||||||
|
if (!value || !data) return null;
|
||||||
|
return addDecimalsFormatNumber(value, data.asset.decimals);
|
||||||
|
},
|
||||||
|
valueFormatter: ({
|
||||||
data,
|
data,
|
||||||
}: VegaValueFormatterParams<AccountFields, 'total'>) =>
|
value,
|
||||||
formatWithAssetDecimals(data, data?.total)
|
}: VegaValueFormatterParams<AccountFields, 'total'>) => {
|
||||||
}
|
if (!data || !value) return '-';
|
||||||
/>
|
return addDecimalsFormatNumberQuantum(
|
||||||
<AgGridColumn
|
value,
|
||||||
colId="accounts-actions"
|
data.asset.decimals,
|
||||||
field="asset.id"
|
data.asset.quantum
|
||||||
{...COL_DEFS.actions}
|
);
|
||||||
minWidth={showDepositButton ? 130 : COL_DEFS.actions.minWidth}
|
},
|
||||||
maxWidth={showDepositButton ? 130 : COL_DEFS.actions.maxWidth}
|
},
|
||||||
cellRenderer={({
|
{
|
||||||
|
colId: 'accounts-actions',
|
||||||
|
field: 'asset.id',
|
||||||
|
...COL_DEFS.actions,
|
||||||
|
minWidth: showDepositButton ? 130 : COL_DEFS.actions.minWidth,
|
||||||
|
maxWidth: showDepositButton ? 130 : COL_DEFS.actions.maxWidth,
|
||||||
|
cellRenderer: ({
|
||||||
value: assetId,
|
value: assetId,
|
||||||
node,
|
node,
|
||||||
}: VegaICellRendererParams<AccountFields, 'asset.id'>) => {
|
}: VegaICellRendererParams<AccountFields, 'asset.id'>) => {
|
||||||
if (!assetId) return null;
|
if (!assetId) return null;
|
||||||
if (node.rowPinned && node.data?.balance === '0') {
|
if (node.rowPinned && node.data?.total === '0') {
|
||||||
return (
|
return (
|
||||||
<CenteredGridCellWrapper className="h-[30px] justify-end py-1">
|
<CenteredGridCellWrapper className="h-[30px] justify-end py-1">
|
||||||
<Button
|
<Button
|
||||||
@ -319,9 +286,42 @@ export const AccountTable = forwardRef<AgGridReact, AccountTableProps>(
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}}
|
},
|
||||||
/>
|
},
|
||||||
</AgGrid>
|
];
|
||||||
|
return defs;
|
||||||
|
}, [
|
||||||
|
onClickAsset,
|
||||||
|
onClickBreakdown,
|
||||||
|
onClickDeposit,
|
||||||
|
onClickWithdraw,
|
||||||
|
props.isReadOnly,
|
||||||
|
showDepositButton,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const data = rowData?.filter(
|
||||||
|
(data) => data.asset.id !== props.pinnedAsset?.id
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<AgGrid
|
||||||
|
{...props}
|
||||||
|
style={{ width: '100%', height: '100%' }}
|
||||||
|
overlayNoRowsTemplate={t('No accounts')}
|
||||||
|
getRowId={({ data }: { data: AccountFields }) => data.asset.id}
|
||||||
|
ref={ref}
|
||||||
|
tooltipShowDelay={500}
|
||||||
|
rowData={data}
|
||||||
|
defaultColDef={{
|
||||||
|
resizable: true,
|
||||||
|
tooltipComponent: TooltipCellComponent,
|
||||||
|
sortable: true,
|
||||||
|
comparator: accountValuesComparator,
|
||||||
|
}}
|
||||||
|
columnDefs={colDefs}
|
||||||
|
getRowHeight={getPinnedAssetRowHeight}
|
||||||
|
pinnedTopRowData={pinnedAsset ? [pinnedAsset] : undefined}
|
||||||
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
@ -35,6 +35,7 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
balance: '100000000',
|
balance: '100000000',
|
||||||
market: null,
|
market: null,
|
||||||
asset: {
|
asset: {
|
||||||
|
// tEURO
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: 'asset-id',
|
id: 'asset-id',
|
||||||
},
|
},
|
||||||
@ -44,6 +45,7 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
type: Schema.AccountType.ACCOUNT_TYPE_GENERAL,
|
type: Schema.AccountType.ACCOUNT_TYPE_GENERAL,
|
||||||
balance: '100000000',
|
balance: '100000000',
|
||||||
asset: {
|
asset: {
|
||||||
|
// tDAI
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: 'asset-id-2',
|
id: 'asset-id-2',
|
||||||
},
|
},
|
||||||
@ -57,6 +59,7 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
id: 'market-2',
|
id: 'market-2',
|
||||||
},
|
},
|
||||||
asset: {
|
asset: {
|
||||||
|
// tEURO
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: 'asset-id',
|
id: 'asset-id',
|
||||||
},
|
},
|
||||||
@ -70,6 +73,7 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
id: 'market-0',
|
id: 'market-0',
|
||||||
},
|
},
|
||||||
asset: {
|
asset: {
|
||||||
|
// AST0
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: 'asset-0',
|
id: 'asset-0',
|
||||||
},
|
},
|
||||||
@ -83,6 +87,7 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
id: 'market-3',
|
id: 'market-3',
|
||||||
},
|
},
|
||||||
asset: {
|
asset: {
|
||||||
|
// AST0
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: 'asset-0',
|
id: 'asset-0',
|
||||||
},
|
},
|
||||||
@ -90,9 +95,10 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
{
|
{
|
||||||
__typename: 'AccountBalance',
|
__typename: 'AccountBalance',
|
||||||
type: Schema.AccountType.ACCOUNT_TYPE_GENERAL,
|
type: Schema.AccountType.ACCOUNT_TYPE_GENERAL,
|
||||||
balance: '10000000000',
|
balance: '10000000',
|
||||||
market: null,
|
market: null,
|
||||||
asset: {
|
asset: {
|
||||||
|
// AST0
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: 'asset-0',
|
id: 'asset-0',
|
||||||
},
|
},
|
||||||
@ -104,6 +110,7 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
balance: '100000001',
|
balance: '100000001',
|
||||||
market: null,
|
market: null,
|
||||||
asset: {
|
asset: {
|
||||||
|
// tBTC (sepolia)
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: 'cee709223217281d7893b650850ae8ee8a18b7539b5658f9b4cc24de95dd18ad',
|
id: 'cee709223217281d7893b650850ae8ee8a18b7539b5658f9b4cc24de95dd18ad',
|
||||||
},
|
},
|
||||||
@ -114,6 +121,7 @@ export const accountFields: AccountFieldsFragment[] = [
|
|||||||
balance: '100000002',
|
balance: '100000002',
|
||||||
market: null,
|
market: null,
|
||||||
asset: {
|
asset: {
|
||||||
|
// tBTC (test)
|
||||||
__typename: 'Asset',
|
__typename: 'Asset',
|
||||||
id: '5cfa87844724df6069b94e4c8a6f03af21907d7bc251593d08e4251043ee9f7c',
|
id: '5cfa87844724df6069b94e4c8a6f03af21907d7bc251593d08e4251043ee9f7c',
|
||||||
},
|
},
|
||||||
|
@ -1,52 +1,106 @@
|
|||||||
import { forwardRef } from 'react';
|
import { forwardRef, useMemo } from 'react';
|
||||||
import { addDecimalsFormatNumber } from '@vegaprotocol/utils';
|
import {
|
||||||
|
addDecimalsFormatNumber,
|
||||||
|
addDecimalsFormatNumberQuantum,
|
||||||
|
} from '@vegaprotocol/utils';
|
||||||
import { t } from '@vegaprotocol/i18n';
|
import { t } from '@vegaprotocol/i18n';
|
||||||
import { Intent } from '@vegaprotocol/ui-toolkit';
|
import { Intent, TooltipCellComponent } from '@vegaprotocol/ui-toolkit';
|
||||||
import { AgGridColumn } from 'ag-grid-react';
|
|
||||||
import type { AgGridReact, AgGridReactProps } from 'ag-grid-react';
|
import type { AgGridReact, AgGridReactProps } from 'ag-grid-react';
|
||||||
import type { AccountFields } from './accounts-data-provider';
|
import type { AccountFields } from './accounts-data-provider';
|
||||||
import { AccountTypeMapping } from '@vegaprotocol/types';
|
import { AccountTypeMapping } from '@vegaprotocol/types';
|
||||||
import type {
|
import type {
|
||||||
ValueProps,
|
VegaICellRendererParams,
|
||||||
VegaValueFormatterParams,
|
VegaValueFormatterParams,
|
||||||
} from '@vegaprotocol/datagrid';
|
} from '@vegaprotocol/datagrid';
|
||||||
import { progressBarCellRendererSelector } from '@vegaprotocol/datagrid';
|
import { ProgressBarCell } from '@vegaprotocol/datagrid';
|
||||||
import { AgGridLazy as AgGrid, PriceCell } from '@vegaprotocol/datagrid';
|
import { AgGridLazy as AgGrid, PriceCell } from '@vegaprotocol/datagrid';
|
||||||
import type { ValueFormatterParams } from 'ag-grid-community';
|
import type { ColDef } from 'ag-grid-community';
|
||||||
import { accountValuesComparator } from './accounts-table';
|
import { accountValuesComparator } from './accounts-table';
|
||||||
|
|
||||||
export const progressBarValueFormatter = ({
|
|
||||||
data,
|
|
||||||
node,
|
|
||||||
}: ValueFormatterParams): ValueProps['valueFormatted'] | undefined => {
|
|
||||||
if (!data || node?.rowPinned) {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
const min = BigInt(data.used);
|
|
||||||
const mid = BigInt(data.available);
|
|
||||||
const max = BigInt(data.total);
|
|
||||||
const range = max > min ? max : min;
|
|
||||||
return {
|
|
||||||
low: addDecimalsFormatNumber(min.toString(), data.asset.decimals),
|
|
||||||
high: addDecimalsFormatNumber(mid.toString(), data.asset.decimals),
|
|
||||||
value: range ? Number((min * BigInt(100)) / range) : 0,
|
|
||||||
intent: Intent.Warning,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
interface BreakdownTableProps extends AgGridReactProps {
|
interface BreakdownTableProps extends AgGridReactProps {
|
||||||
data: AccountFields[] | null;
|
data: AccountFields[] | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const BreakdownTable = forwardRef<AgGridReact, BreakdownTableProps>(
|
const BreakdownTable = forwardRef<AgGridReact, BreakdownTableProps>(
|
||||||
({ data }, ref) => {
|
({ data }, ref) => {
|
||||||
|
const coldefs = useMemo(() => {
|
||||||
|
const defs: ColDef[] = [
|
||||||
|
{
|
||||||
|
headerName: t('Market'),
|
||||||
|
field: 'market.tradableInstrument.instrument.name',
|
||||||
|
valueFormatter: ({
|
||||||
|
value,
|
||||||
|
}: VegaValueFormatterParams<
|
||||||
|
AccountFields,
|
||||||
|
'market.tradableInstrument.instrument.name'
|
||||||
|
>) => {
|
||||||
|
if (!value) return 'None';
|
||||||
|
return value;
|
||||||
|
},
|
||||||
|
minWidth: 200,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headerName: t('Account type'),
|
||||||
|
field: 'type',
|
||||||
|
maxWidth: 300,
|
||||||
|
valueFormatter: ({
|
||||||
|
value,
|
||||||
|
}: VegaValueFormatterParams<AccountFields, 'type'>) => {
|
||||||
|
return value
|
||||||
|
? AccountTypeMapping[value as keyof typeof AccountTypeMapping]
|
||||||
|
: '';
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headerName: t('Balance'),
|
||||||
|
field: 'used',
|
||||||
|
flex: 2,
|
||||||
|
maxWidth: 500,
|
||||||
|
type: 'rightAligned',
|
||||||
|
tooltipComponent: TooltipCellComponent,
|
||||||
|
tooltipValueGetter: ({ value, data }) => {
|
||||||
|
return addDecimalsFormatNumber(value, data.asset.decimals);
|
||||||
|
},
|
||||||
|
cellRenderer: ({
|
||||||
|
data,
|
||||||
|
node,
|
||||||
|
}: VegaICellRendererParams<AccountFields, 'used'>) => {
|
||||||
|
if (!data || node?.rowPinned) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
const min = BigInt(data.used);
|
||||||
|
const mid = BigInt(data.available);
|
||||||
|
const max = BigInt(data.total);
|
||||||
|
const range = max > min ? max : min;
|
||||||
|
const formattedData = {
|
||||||
|
low: addDecimalsFormatNumberQuantum(
|
||||||
|
min.toString(),
|
||||||
|
data.asset.decimals,
|
||||||
|
data.asset.quantum
|
||||||
|
),
|
||||||
|
high: addDecimalsFormatNumberQuantum(
|
||||||
|
mid.toString(),
|
||||||
|
data.asset.decimals,
|
||||||
|
data.asset.quantum
|
||||||
|
),
|
||||||
|
value: range ? Number((min * BigInt(100)) / range) : 0,
|
||||||
|
intent: Intent.Warning,
|
||||||
|
};
|
||||||
|
return <ProgressBarCell valueFormatted={formattedData} />;
|
||||||
|
},
|
||||||
|
comparator: accountValuesComparator,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
return defs;
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AgGrid
|
<AgGrid
|
||||||
style={{ width: '100%', height: '100%' }}
|
style={{ width: '100%', height: '100%' }}
|
||||||
overlayNoRowsTemplate={t('Collateral not used')}
|
overlayNoRowsTemplate={t('Collateral not used')}
|
||||||
rowData={data}
|
rowData={data}
|
||||||
getRowId={({ data }: { data: AccountFields }) =>
|
getRowId={({ data }: { data: AccountFields }) =>
|
||||||
`${data.asset.id}-${data.type}-${data.market?.id}`
|
`${data.asset.id}:${data.type}:${data.market?.id}`
|
||||||
}
|
}
|
||||||
ref={ref}
|
ref={ref}
|
||||||
rowHeight={34}
|
rowHeight={34}
|
||||||
@ -57,44 +111,8 @@ const BreakdownTable = forwardRef<AgGridReact, BreakdownTableProps>(
|
|||||||
resizable: true,
|
resizable: true,
|
||||||
sortable: true,
|
sortable: true,
|
||||||
}}
|
}}
|
||||||
>
|
columnDefs={coldefs}
|
||||||
<AgGridColumn
|
/>
|
||||||
headerName={t('Market')}
|
|
||||||
field="market.tradableInstrument.instrument.name"
|
|
||||||
valueFormatter={({
|
|
||||||
value,
|
|
||||||
}: VegaValueFormatterParams<
|
|
||||||
AccountFields,
|
|
||||||
'market.tradableInstrument.instrument.name'
|
|
||||||
>) => {
|
|
||||||
if (!value) return 'None';
|
|
||||||
return value;
|
|
||||||
}}
|
|
||||||
minWidth={200}
|
|
||||||
/>
|
|
||||||
<AgGridColumn
|
|
||||||
headerName={t('Account type')}
|
|
||||||
field="type"
|
|
||||||
maxWidth={300}
|
|
||||||
valueFormatter={({
|
|
||||||
value,
|
|
||||||
}: VegaValueFormatterParams<AccountFields, 'type'>) =>
|
|
||||||
value
|
|
||||||
? AccountTypeMapping[value as keyof typeof AccountTypeMapping]
|
|
||||||
: ''
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<AgGridColumn
|
|
||||||
headerName={t('Balance')}
|
|
||||||
field="used"
|
|
||||||
flex={2}
|
|
||||||
maxWidth={500}
|
|
||||||
cellRendererSelector={progressBarCellRendererSelector}
|
|
||||||
valueFormatter={progressBarValueFormatter}
|
|
||||||
comparator={accountValuesComparator}
|
|
||||||
/>
|
|
||||||
</AgGrid>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
@ -26,6 +26,7 @@ export const AgGridThemed = ({
|
|||||||
enableCellTextSelection: true,
|
enableCellTextSelection: true,
|
||||||
overlayLoadingTemplate: t('Loading...'),
|
overlayLoadingTemplate: t('Loading...'),
|
||||||
overlayNoRowsTemplate: t('No data'),
|
overlayNoRowsTemplate: t('No data'),
|
||||||
|
suppressCellFocus: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
const wrapperClasses = classNames('vega-ag-grid', {
|
const wrapperClasses = classNames('vega-ag-grid', {
|
||||||
|
@ -1,9 +1,5 @@
|
|||||||
import type { Intent } from '@vegaprotocol/ui-toolkit';
|
import type { Intent } from '@vegaprotocol/ui-toolkit';
|
||||||
import { ProgressBar } from '@vegaprotocol/ui-toolkit';
|
import { ProgressBar } from '@vegaprotocol/ui-toolkit';
|
||||||
import type {
|
|
||||||
CellRendererSelectorResult,
|
|
||||||
ICellRendererParams,
|
|
||||||
} from 'ag-grid-community';
|
|
||||||
|
|
||||||
export interface ValueProps {
|
export interface ValueProps {
|
||||||
valueFormatted?: {
|
valueFormatted?: {
|
||||||
@ -19,7 +15,7 @@ export const EmptyCell = () => '';
|
|||||||
export const ProgressBarCell = ({ valueFormatted }: ValueProps) => {
|
export const ProgressBarCell = ({ valueFormatted }: ValueProps) => {
|
||||||
return valueFormatted ? (
|
return valueFormatted ? (
|
||||||
<>
|
<>
|
||||||
<div className="flex justify-between leading-tight font-mono">
|
<div className="text-right leading-tight font-mono">
|
||||||
<div>
|
<div>
|
||||||
{valueFormatted.low} ({valueFormatted.value}%)
|
{valueFormatted.low} ({valueFormatted.value}%)
|
||||||
</div>
|
</div>
|
||||||
@ -32,11 +28,3 @@ export const ProgressBarCell = ({ valueFormatted }: ValueProps) => {
|
|||||||
</>
|
</>
|
||||||
) : null;
|
) : null;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const progressBarCellRendererSelector = (
|
|
||||||
params: ICellRendererParams
|
|
||||||
): CellRendererSelectorResult => {
|
|
||||||
return {
|
|
||||||
component: ProgressBarCell,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
@ -9,6 +9,8 @@ import {
|
|||||||
} from '@radix-ui/react-tooltip';
|
} from '@radix-ui/react-tooltip';
|
||||||
import type { ITooltipParams } from 'ag-grid-community';
|
import type { ITooltipParams } from 'ag-grid-community';
|
||||||
|
|
||||||
|
const tooltipContentClasses =
|
||||||
|
'max-w-sm bg-vega-light-100 dark:bg-vega-dark-100 border border-vega-light-200 dark:border-vega-dark-200 px-2 py-1 z-20 rounded text-xs break-word';
|
||||||
export interface TooltipProps {
|
export interface TooltipProps {
|
||||||
children: React.ReactElement;
|
children: React.ReactElement;
|
||||||
description?: string | ReactNode;
|
description?: string | ReactNode;
|
||||||
@ -40,7 +42,7 @@ export const Tooltip = ({
|
|||||||
align={align}
|
align={align}
|
||||||
side={side}
|
side={side}
|
||||||
alignOffset={8}
|
alignOffset={8}
|
||||||
className="max-w-sm border border-neutral-600 bg-neutral-100 dark:bg-neutral-800 px-4 py-2 z-20 rounded text-sm text-black dark:text-white break-word"
|
className={tooltipContentClasses}
|
||||||
>
|
>
|
||||||
<div className="relative z-0" data-testid="tooltip-content">
|
<div className="relative z-0" data-testid="tooltip-content">
|
||||||
{description}
|
{description}
|
||||||
@ -55,9 +57,5 @@ export const Tooltip = ({
|
|||||||
);
|
);
|
||||||
|
|
||||||
export const TooltipCellComponent = (props: ITooltipParams) => {
|
export const TooltipCellComponent = (props: ITooltipParams) => {
|
||||||
return (
|
return <p className={tooltipContentClasses}>{props.value}</p>;
|
||||||
<p className="max-w-sm border border-neutral-600 bg-neutral-100 dark:bg-neutral-800 px-4 py-2 z-20 rounded text-sm break-word text-black dark:text-white">
|
|
||||||
{props.value}
|
|
||||||
</p>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user