2023-06-05 10:40:34 +00:00
|
|
|
import { useRef, memo, useCallback, useState, useEffect } from 'react';
|
|
|
|
import { addDecimalsFormatNumber } from '@vegaprotocol/utils';
|
2023-02-28 18:56:29 +00:00
|
|
|
import { t } from '@vegaprotocol/i18n';
|
2023-05-09 08:04:33 +00:00
|
|
|
import { useBottomPlaceholder } from '@vegaprotocol/datagrid';
|
2023-05-08 01:01:07 +00:00
|
|
|
import { useDataProvider } from '@vegaprotocol/data-provider';
|
2022-04-06 17:48:05 +00:00
|
|
|
import { AsyncRenderer } from '@vegaprotocol/ui-toolkit';
|
|
|
|
import type { AgGridReact } from 'ag-grid-react';
|
2023-06-05 10:40:34 +00:00
|
|
|
import type { RowDataUpdatedEvent } from 'ag-grid-community';
|
|
|
|
import type { AccountFields } from './accounts-data-provider';
|
|
|
|
import {
|
|
|
|
aggregatedAccountsDataProvider,
|
|
|
|
aggregatedAccountDataProvider,
|
|
|
|
} from './accounts-data-provider';
|
2023-02-17 13:32:20 +00:00
|
|
|
import type { PinnedAsset } from './accounts-table';
|
2022-09-30 00:40:44 +00:00
|
|
|
import { AccountTable } from './accounts-table';
|
2023-06-05 10:40:34 +00:00
|
|
|
import isEqual from 'lodash/isEqual';
|
|
|
|
import { Dialog } from '@vegaprotocol/ui-toolkit';
|
|
|
|
import BreakdownTable from './breakdown-table';
|
|
|
|
|
|
|
|
const AccountBreakdown = ({
|
|
|
|
assetId,
|
|
|
|
partyId,
|
|
|
|
}: {
|
|
|
|
assetId: string;
|
|
|
|
partyId: string;
|
|
|
|
}) => {
|
|
|
|
const { data } = useDataProvider({
|
|
|
|
dataProvider: aggregatedAccountDataProvider,
|
|
|
|
variables: { partyId, assetId },
|
|
|
|
});
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className="h-[35vh] w-full m-auto flex flex-col"
|
|
|
|
data-testid="usage-breakdown"
|
|
|
|
>
|
|
|
|
<h1 className="text-xl mb-4">
|
|
|
|
{data?.asset?.symbol} {t('usage breakdown')}
|
|
|
|
</h1>
|
|
|
|
{data && (
|
|
|
|
<p className="mb-2 text-sm">
|
|
|
|
{t('You have %s %s in total.', [
|
|
|
|
addDecimalsFormatNumber(data.total, data.asset.decimals),
|
|
|
|
data.asset.symbol,
|
|
|
|
])}
|
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
<BreakdownTable data={data?.breakdown || null} domLayout="autoHeight" />
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
2022-04-06 17:48:05 +00:00
|
|
|
|
2022-09-30 00:40:44 +00:00
|
|
|
interface AccountManagerProps {
|
2022-04-06 17:48:05 +00:00
|
|
|
partyId: string;
|
2022-11-23 23:42:22 +00:00
|
|
|
onClickAsset: (assetId: string) => void;
|
2022-09-30 00:40:44 +00:00
|
|
|
onClickWithdraw?: (assetId?: string) => void;
|
|
|
|
onClickDeposit?: (assetId?: string) => void;
|
2023-01-31 16:04:52 +00:00
|
|
|
isReadOnly: boolean;
|
2023-02-17 13:32:20 +00:00
|
|
|
pinnedAsset?: PinnedAsset;
|
2023-03-17 09:50:43 +00:00
|
|
|
noBottomPlaceholder?: boolean;
|
2023-05-09 06:09:53 +00:00
|
|
|
storeKey?: string;
|
2022-04-06 17:48:05 +00:00
|
|
|
}
|
|
|
|
|
2022-12-02 12:32:54 +00:00
|
|
|
export const AccountManager = ({
|
|
|
|
onClickAsset,
|
|
|
|
onClickWithdraw,
|
|
|
|
onClickDeposit,
|
|
|
|
partyId,
|
2023-01-31 16:04:52 +00:00
|
|
|
isReadOnly,
|
2023-02-17 13:32:20 +00:00
|
|
|
pinnedAsset,
|
2023-03-17 09:50:43 +00:00
|
|
|
noBottomPlaceholder,
|
2023-05-09 06:09:53 +00:00
|
|
|
storeKey,
|
2022-12-02 12:32:54 +00:00
|
|
|
}: AccountManagerProps) => {
|
|
|
|
const gridRef = useRef<AgGridReact | null>(null);
|
2023-06-05 10:40:34 +00:00
|
|
|
const [hasData, setHasData] = useState(Boolean(pinnedAsset));
|
|
|
|
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]
|
|
|
|
);
|
2023-03-09 10:03:50 +00:00
|
|
|
const { data, loading, error, reload } = useDataProvider({
|
2022-12-02 12:32:54 +00:00
|
|
|
dataProvider: aggregatedAccountsDataProvider,
|
2023-06-05 10:40:34 +00:00
|
|
|
variables: { partyId },
|
|
|
|
update,
|
2022-12-02 12:32:54 +00:00
|
|
|
});
|
2023-05-25 10:59:08 +00:00
|
|
|
const bottomPlaceholderProps = useBottomPlaceholder({
|
2023-03-08 14:58:40 +00:00
|
|
|
gridRef,
|
2023-03-17 09:50:43 +00:00
|
|
|
disabled: noBottomPlaceholder,
|
2023-03-08 14:58:40 +00:00
|
|
|
});
|
|
|
|
|
2023-06-05 10:40:34 +00:00
|
|
|
useEffect(
|
|
|
|
() => setHasData(Boolean(pinnedAsset || data?.length)),
|
|
|
|
[data, pinnedAsset]
|
|
|
|
);
|
|
|
|
|
|
|
|
const onRowDataUpdated = useCallback(
|
|
|
|
(event: RowDataUpdatedEvent) => {
|
|
|
|
setHasData(Boolean(pinnedAsset || event.api?.getModel().getRowCount()));
|
|
|
|
},
|
|
|
|
[pinnedAsset]
|
|
|
|
);
|
|
|
|
|
2022-12-02 12:32:54 +00:00
|
|
|
return (
|
|
|
|
<div className="relative h-full">
|
|
|
|
<AccountTable
|
|
|
|
ref={gridRef}
|
2023-02-16 16:30:32 +00:00
|
|
|
rowData={error ? [] : data}
|
2022-12-02 12:32:54 +00:00
|
|
|
onClickAsset={onClickAsset}
|
|
|
|
onClickDeposit={onClickDeposit}
|
|
|
|
onClickWithdraw={onClickWithdraw}
|
2023-06-05 10:40:34 +00:00
|
|
|
onClickBreakdown={setBreakdownAssetId}
|
|
|
|
onRowDataUpdated={onRowDataUpdated}
|
2023-01-31 16:04:52 +00:00
|
|
|
isReadOnly={isReadOnly}
|
2023-03-15 15:08:48 +00:00
|
|
|
suppressLoadingOverlay
|
|
|
|
suppressNoRowsOverlay
|
2023-02-17 13:32:20 +00:00
|
|
|
pinnedAsset={pinnedAsset}
|
2023-05-09 06:09:53 +00:00
|
|
|
storeKey={storeKey}
|
2023-03-08 14:58:40 +00:00
|
|
|
{...bottomPlaceholderProps}
|
2022-12-02 12:32:54 +00:00
|
|
|
/>
|
2023-01-11 14:43:52 +00:00
|
|
|
<div className="pointer-events-none absolute inset-0">
|
2022-12-02 12:32:54 +00:00
|
|
|
<AsyncRenderer
|
2023-01-11 14:43:52 +00:00
|
|
|
data={data}
|
2023-06-05 10:40:34 +00:00
|
|
|
noDataCondition={() => !hasData}
|
2022-12-02 12:32:54 +00:00
|
|
|
error={error}
|
|
|
|
loading={loading}
|
2023-02-17 13:32:20 +00:00
|
|
|
noDataMessage={pinnedAsset ? ' ' : t('No accounts')}
|
2023-02-16 16:30:32 +00:00
|
|
|
reload={reload}
|
2022-11-07 12:14:21 +00:00
|
|
|
/>
|
2022-12-02 12:32:54 +00:00
|
|
|
</div>
|
2023-06-05 10:40:34 +00:00
|
|
|
<Dialog
|
|
|
|
size="medium"
|
|
|
|
open={Boolean(breakdownAssetId)}
|
|
|
|
onChange={(isOpen) => {
|
|
|
|
if (!isOpen) {
|
|
|
|
setBreakdownAssetId(undefined);
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{breakdownAssetId && (
|
|
|
|
<AccountBreakdown assetId={breakdownAssetId} partyId={partyId} />
|
|
|
|
)}
|
|
|
|
</Dialog>
|
2022-12-02 12:32:54 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
2022-09-30 00:40:44 +00:00
|
|
|
|
2022-11-07 12:14:21 +00:00
|
|
|
export default memo(AccountManager);
|