feat(explorer): referral and team transaction support (#5623)
This commit is contained in:
parent
e16c447564
commit
0da20b750f
54
apps/explorer/src/app/components/txs/details/referrals/__generated__/code-owner.ts
generated
Normal file
54
apps/explorer/src/app/components/txs/details/referrals/__generated__/code-owner.ts
generated
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
import * as Types from '@vegaprotocol/types';
|
||||||
|
|
||||||
|
import { gql } from '@apollo/client';
|
||||||
|
import * as Apollo from '@apollo/client';
|
||||||
|
const defaultOptions = {} as const;
|
||||||
|
export type ExplorerReferralCodeOwnerQueryVariables = Types.Exact<{
|
||||||
|
id: Types.Scalars['ID'];
|
||||||
|
}>;
|
||||||
|
|
||||||
|
|
||||||
|
export type ExplorerReferralCodeOwnerQuery = { __typename?: 'Query', referralSets: { __typename?: 'ReferralSetConnection', edges: Array<{ __typename?: 'ReferralSetEdge', node: { __typename?: 'ReferralSet', createdAt: any, updatedAt: any, referrer: string } } | null> } };
|
||||||
|
|
||||||
|
|
||||||
|
export const ExplorerReferralCodeOwnerDocument = gql`
|
||||||
|
query ExplorerReferralCodeOwner($id: ID!) {
|
||||||
|
referralSets(id: $id) {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
referrer
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* __useExplorerReferralCodeOwnerQuery__
|
||||||
|
*
|
||||||
|
* To run a query within a React component, call `useExplorerReferralCodeOwnerQuery` and pass it any options that fit your needs.
|
||||||
|
* When your component renders, `useExplorerReferralCodeOwnerQuery` returns an object from Apollo Client that contains loading, error, and data properties
|
||||||
|
* you can use to render your UI.
|
||||||
|
*
|
||||||
|
* @param baseOptions options that will be passed into the query, supported options are listed on: https://www.apollographql.com/docs/react/api/react-hooks/#options;
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* const { data, loading, error } = useExplorerReferralCodeOwnerQuery({
|
||||||
|
* variables: {
|
||||||
|
* id: // value for 'id'
|
||||||
|
* },
|
||||||
|
* });
|
||||||
|
*/
|
||||||
|
export function useExplorerReferralCodeOwnerQuery(baseOptions: Apollo.QueryHookOptions<ExplorerReferralCodeOwnerQuery, ExplorerReferralCodeOwnerQueryVariables>) {
|
||||||
|
const options = {...defaultOptions, ...baseOptions}
|
||||||
|
return Apollo.useQuery<ExplorerReferralCodeOwnerQuery, ExplorerReferralCodeOwnerQueryVariables>(ExplorerReferralCodeOwnerDocument, options);
|
||||||
|
}
|
||||||
|
export function useExplorerReferralCodeOwnerLazyQuery(baseOptions?: Apollo.LazyQueryHookOptions<ExplorerReferralCodeOwnerQuery, ExplorerReferralCodeOwnerQueryVariables>) {
|
||||||
|
const options = {...defaultOptions, ...baseOptions}
|
||||||
|
return Apollo.useLazyQuery<ExplorerReferralCodeOwnerQuery, ExplorerReferralCodeOwnerQueryVariables>(ExplorerReferralCodeOwnerDocument, options);
|
||||||
|
}
|
||||||
|
export type ExplorerReferralCodeOwnerQueryHookResult = ReturnType<typeof useExplorerReferralCodeOwnerQuery>;
|
||||||
|
export type ExplorerReferralCodeOwnerLazyQueryHookResult = ReturnType<typeof useExplorerReferralCodeOwnerLazyQuery>;
|
||||||
|
export type ExplorerReferralCodeOwnerQueryResult = Apollo.QueryResult<ExplorerReferralCodeOwnerQuery, ExplorerReferralCodeOwnerQueryVariables>;
|
@ -0,0 +1,11 @@
|
|||||||
|
query ExplorerReferralCodeOwner($id: ID!) {
|
||||||
|
referralSets(id: $id) {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
referrer
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,95 @@
|
|||||||
|
import { render, waitFor } from '@testing-library/react';
|
||||||
|
import type { MockedResponse } from '@apollo/client/testing';
|
||||||
|
import { MockedProvider } from '@apollo/client/testing';
|
||||||
|
import { MemoryRouter } from 'react-router-dom';
|
||||||
|
import { ReferralCodeOwner } from './referral-code-owner';
|
||||||
|
import type { ReferralCodeOwnerProps } from './referral-code-owner';
|
||||||
|
import { ExplorerReferralCodeOwnerDocument } from './__generated__/code-owner';
|
||||||
|
const renderComponent = (
|
||||||
|
props: ReferralCodeOwnerProps,
|
||||||
|
mocks: MockedResponse[]
|
||||||
|
) => {
|
||||||
|
return render(
|
||||||
|
<MockedProvider mocks={mocks}>
|
||||||
|
<MemoryRouter>
|
||||||
|
<ReferralCodeOwner {...props} />
|
||||||
|
</MemoryRouter>
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
describe('ReferralCodeOwner', () => {
|
||||||
|
it('should render loading state', () => {
|
||||||
|
const mocks = [
|
||||||
|
{
|
||||||
|
request: {
|
||||||
|
query: ExplorerReferralCodeOwnerDocument,
|
||||||
|
variables: {
|
||||||
|
id: 'ABC123',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const { getByText } = renderComponent({ code: 'ABC123' }, mocks);
|
||||||
|
|
||||||
|
expect(getByText('Loading...')).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render error state', async () => {
|
||||||
|
const errorMessage = 'Error fetching referrer: ABC123';
|
||||||
|
const mocks = [
|
||||||
|
{
|
||||||
|
request: {
|
||||||
|
query: ExplorerReferralCodeOwnerDocument,
|
||||||
|
variables: {
|
||||||
|
id: 'ABC123',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
error: new Error('nope'),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const { getByText } = renderComponent({ code: 'ABC123' }, mocks);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(getByText(errorMessage)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render link to referring party', async () => {
|
||||||
|
const referrerId = 'DEF789';
|
||||||
|
|
||||||
|
const mocks = [
|
||||||
|
{
|
||||||
|
request: {
|
||||||
|
query: ExplorerReferralCodeOwnerDocument,
|
||||||
|
variables: {
|
||||||
|
id: 'ABC123',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
result: {
|
||||||
|
data: {
|
||||||
|
referralSets: {
|
||||||
|
edges: [
|
||||||
|
{
|
||||||
|
node: {
|
||||||
|
__typename: 'ReferralSet',
|
||||||
|
referrer: referrerId,
|
||||||
|
createdAt: '123',
|
||||||
|
updatedAt: '456',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const { getByText } = renderComponent({ code: 'ABC123' }, mocks);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(getByText(referrerId)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,26 @@
|
|||||||
|
import { TableCell } from '../../../table';
|
||||||
|
import { useExplorerReferralCodeOwnerQuery } from './__generated__/code-owner';
|
||||||
|
import { PartyLink } from '../../../links';
|
||||||
|
|
||||||
|
export interface ReferralCodeOwnerProps {
|
||||||
|
code: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render the owner of a referral code
|
||||||
|
*/
|
||||||
|
export const ReferralCodeOwner = ({ code }: ReferralCodeOwnerProps) => {
|
||||||
|
const { data, error, loading } = useExplorerReferralCodeOwnerQuery({
|
||||||
|
variables: {
|
||||||
|
id: code,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
const referrer = data?.referralSets.edges[0]?.node.referrer || '';
|
||||||
|
return (
|
||||||
|
<TableCell>
|
||||||
|
{loading && 'Loading...'}
|
||||||
|
{error && `Error fetching referrer: ${code}`}
|
||||||
|
{referrer.length > 0 && <PartyLink id={referrer} />}
|
||||||
|
</TableCell>
|
||||||
|
);
|
||||||
|
};
|
@ -0,0 +1,93 @@
|
|||||||
|
import { render } from '@testing-library/react';
|
||||||
|
import { ReferralTeam } from './team';
|
||||||
|
import type { CreateReferralSet } from './team';
|
||||||
|
import { MockedProvider } from '@apollo/client/testing';
|
||||||
|
|
||||||
|
describe('ReferralTeam', () => {
|
||||||
|
const team = {
|
||||||
|
name: 'Test Team',
|
||||||
|
teamUrl: 'https://example.com/team',
|
||||||
|
avatarUrl: 'https://example.com/avatar',
|
||||||
|
closed: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockTx: CreateReferralSet = {
|
||||||
|
team,
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockId = '123456';
|
||||||
|
const mockCreator = 'JohnDoe';
|
||||||
|
|
||||||
|
it('should render the team name', () => {
|
||||||
|
const { getByText } = render(
|
||||||
|
<MockedProvider>
|
||||||
|
<ReferralTeam tx={mockTx} id={mockId} creator={mockCreator} />
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
expect(getByText('Test Team')).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render the team ID', () => {
|
||||||
|
const { getByText } = render(
|
||||||
|
<MockedProvider>
|
||||||
|
<ReferralTeam tx={mockTx} id={mockId} creator={mockCreator} />
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
expect(getByText('Id')).toBeInTheDocument();
|
||||||
|
expect(getByText(mockId)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render the creator', () => {
|
||||||
|
const { getByText } = render(
|
||||||
|
<MockedProvider>
|
||||||
|
<ReferralTeam tx={mockTx} id={mockId} creator={mockCreator} />
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
expect(getByText('Creator')).toBeInTheDocument();
|
||||||
|
expect(getByText(mockCreator)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render the team URL', () => {
|
||||||
|
const { getByText } = render(
|
||||||
|
<MockedProvider>
|
||||||
|
<ReferralTeam tx={mockTx} id={mockId} creator={mockCreator} />
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
expect(getByText('Team URL')).toBeInTheDocument();
|
||||||
|
expect(getByText(team.teamUrl)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render the avatar URL', () => {
|
||||||
|
const { getByText } = render(
|
||||||
|
<MockedProvider>
|
||||||
|
<ReferralTeam tx={mockTx} id={mockId} creator={mockCreator} />
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
expect(getByText('Avatar')).toBeInTheDocument();
|
||||||
|
expect(getByText(team.avatarUrl)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render the open status as a tick if closed is falsy', () => {
|
||||||
|
const { getByTestId } = render(
|
||||||
|
<MockedProvider>
|
||||||
|
<ReferralTeam tx={mockTx} id={mockId} creator={mockCreator} />
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
expect(getByTestId('open-yes')).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render the open status as a cross if it is truthy', () => {
|
||||||
|
const m = {
|
||||||
|
team: {
|
||||||
|
closed: true,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const { getByTestId } = render(
|
||||||
|
<MockedProvider>
|
||||||
|
<ReferralTeam tx={m} id={mockId} creator={mockCreator} />
|
||||||
|
</MockedProvider>
|
||||||
|
);
|
||||||
|
expect(getByTestId('open-no')).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,73 @@
|
|||||||
|
import {
|
||||||
|
VegaIcon,
|
||||||
|
Icon,
|
||||||
|
KeyValueTable,
|
||||||
|
KeyValueTableRow,
|
||||||
|
VegaIconNames,
|
||||||
|
} from '@vegaprotocol/ui-toolkit';
|
||||||
|
import type { components } from '../../../../../types/explorer';
|
||||||
|
import Hash from '../../../links/hash';
|
||||||
|
import { t } from 'i18next';
|
||||||
|
import { PartyLink } from '../../../links';
|
||||||
|
|
||||||
|
export type CreateReferralSet = components['schemas']['v1CreateReferralSet'];
|
||||||
|
export type ReferralTeam = CreateReferralSet['team'];
|
||||||
|
|
||||||
|
export interface ReferralTeamProps {
|
||||||
|
tx: CreateReferralSet;
|
||||||
|
id: string;
|
||||||
|
creator: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Renders the details for a team in a CreateReferralSet or UpdateReferralSet transaction.
|
||||||
|
*
|
||||||
|
* Intentionally does not render the avatar image or link to the team url.
|
||||||
|
*/
|
||||||
|
export const ReferralTeam = ({ tx, id, creator }: ReferralTeamProps) => {
|
||||||
|
if (!tx.team) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<section>
|
||||||
|
<div className="inline-block mr-2 leading-none">
|
||||||
|
<VegaIcon name={VegaIconNames.TEAM} />
|
||||||
|
</div>
|
||||||
|
{tx.team.name && (
|
||||||
|
<h3 className="inline-block leading-loose">{tx.team.name}</h3>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className="min-w-fit max-w-2xl block">
|
||||||
|
<KeyValueTable>
|
||||||
|
<KeyValueTableRow>
|
||||||
|
{t('Id')}
|
||||||
|
<Hash text={id} truncate={false} />
|
||||||
|
</KeyValueTableRow>
|
||||||
|
<KeyValueTableRow>
|
||||||
|
{t('Creator')}
|
||||||
|
<PartyLink id={creator} truncate={false} />
|
||||||
|
</KeyValueTableRow>
|
||||||
|
{tx.team.teamUrl && (
|
||||||
|
<KeyValueTableRow>
|
||||||
|
{t('Team URL')}
|
||||||
|
<Hash text={tx.team.teamUrl} truncate={false} />
|
||||||
|
</KeyValueTableRow>
|
||||||
|
)}
|
||||||
|
{tx.team.avatarUrl && (
|
||||||
|
<KeyValueTableRow>
|
||||||
|
{t('Avatar')}
|
||||||
|
<Hash text={tx.team.avatarUrl} truncate={false} />
|
||||||
|
</KeyValueTableRow>
|
||||||
|
)}
|
||||||
|
<KeyValueTableRow>
|
||||||
|
{t('Open')}
|
||||||
|
<span data-testid={!tx.team.closed ? 'open-yes' : 'open-no'}>
|
||||||
|
{!tx.team.closed ? <Icon name="tick" /> : <Icon name="cross" />}
|
||||||
|
</span>
|
||||||
|
</KeyValueTableRow>
|
||||||
|
</KeyValueTable>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
);
|
||||||
|
};
|
@ -27,7 +27,8 @@ export const sharedHeaderProps = {
|
|||||||
className: 'align-top',
|
className: 'align-top',
|
||||||
};
|
};
|
||||||
|
|
||||||
const Labels: Record<BlockExplorerTransactionResult['type'], string> = {
|
// The incoming type field is usually the right thing to show. Exceptions are listed here
|
||||||
|
const LabelOverrides: Record<BlockExplorerTransactionResult['type'], string> = {
|
||||||
'Stop Orders Submission': 'Stop Order',
|
'Stop Orders Submission': 'Stop Order',
|
||||||
'Stop Orders Cancellation': 'Cancel Stop Order',
|
'Stop Orders Cancellation': 'Cancel Stop Order',
|
||||||
};
|
};
|
||||||
@ -50,7 +51,7 @@ export const TxDetailsShared = ({
|
|||||||
const time: string = blockData?.result.block.header.time || '';
|
const time: string = blockData?.result.block.header.time || '';
|
||||||
const height: string = blockData?.result.block.header.height || txData.block;
|
const height: string = blockData?.result.block.header.height || txData.block;
|
||||||
|
|
||||||
const type = Labels[txData.type] || txData.type;
|
const type = LabelOverrides[txData.type] || txData.type;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
@ -0,0 +1,47 @@
|
|||||||
|
import { t } from '@vegaprotocol/i18n';
|
||||||
|
import type { BlockExplorerTransactionResult } from '../../../routes/types/block-explorer-response';
|
||||||
|
import type { TendermintBlocksResponse } from '../../../routes/blocks/tendermint-blocks-response';
|
||||||
|
import { TxDetailsShared } from './shared/tx-details-shared';
|
||||||
|
import { TableCell, TableRow, TableWithTbody } from '../../table';
|
||||||
|
import { ReferralCodeOwner } from './referrals/referral-code-owner';
|
||||||
|
|
||||||
|
interface TxDetailsApplyReferralCodeProps {
|
||||||
|
txData: BlockExplorerTransactionResult | undefined;
|
||||||
|
pubKey: string | undefined;
|
||||||
|
blockData: TendermintBlocksResponse | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The signature can be turned in to an id with txSignatureToDeterministicId but
|
||||||
|
*/
|
||||||
|
export const TxDetailsApplyReferralCode = ({
|
||||||
|
txData,
|
||||||
|
pubKey,
|
||||||
|
blockData,
|
||||||
|
}: TxDetailsApplyReferralCodeProps) => {
|
||||||
|
if (!txData || !txData.command.applyReferralCode || !txData.signature) {
|
||||||
|
return <>{t('Awaiting Block Explorer transaction details')}</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const referralCode = txData.command.applyReferralCode.id;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<TableWithTbody className="mb-8" allowWrap={true}>
|
||||||
|
<TxDetailsShared
|
||||||
|
txData={txData}
|
||||||
|
pubKey={pubKey}
|
||||||
|
blockData={blockData}
|
||||||
|
/>
|
||||||
|
<TableRow modifier="bordered">
|
||||||
|
<TableCell>{t('Applied Code')}</TableCell>
|
||||||
|
<TableCell>{referralCode}</TableCell>
|
||||||
|
</TableRow>
|
||||||
|
<TableRow modifier="bordered">
|
||||||
|
<TableCell>{t('Referrer')}</TableCell>
|
||||||
|
<ReferralCodeOwner code={referralCode} />
|
||||||
|
</TableRow>
|
||||||
|
</TableWithTbody>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
@ -0,0 +1,52 @@
|
|||||||
|
import { t } from '@vegaprotocol/i18n';
|
||||||
|
import type { BlockExplorerTransactionResult } from '../../../routes/types/block-explorer-response';
|
||||||
|
import type { TendermintBlocksResponse } from '../../../routes/blocks/tendermint-blocks-response';
|
||||||
|
import { TxDetailsShared } from './shared/tx-details-shared';
|
||||||
|
import { TableCell, TableRow, TableWithTbody } from '../../table';
|
||||||
|
import { txSignatureToDeterministicId } from '../lib/deterministic-ids';
|
||||||
|
import { ReferralTeam } from './referrals/team';
|
||||||
|
|
||||||
|
interface TxDetailsCreateReferralProps {
|
||||||
|
txData: BlockExplorerTransactionResult | undefined;
|
||||||
|
pubKey: string | undefined;
|
||||||
|
blockData: TendermintBlocksResponse | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The signature can be turned in to an id with txSignatureToDeterministicId but
|
||||||
|
*/
|
||||||
|
export const TxDetailsCreateReferralSet = ({
|
||||||
|
txData,
|
||||||
|
pubKey,
|
||||||
|
blockData,
|
||||||
|
}: TxDetailsCreateReferralProps) => {
|
||||||
|
if (!txData || !txData.command.createReferralSet || !txData.signature) {
|
||||||
|
return <>{t('Awaiting Block Explorer transaction details')}</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const id = txSignatureToDeterministicId(txData.signature.value);
|
||||||
|
|
||||||
|
const isTeam = txData.command.createReferralSet.isTeam || false;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<TableWithTbody className="mb-8" allowWrap={true}>
|
||||||
|
<TxDetailsShared
|
||||||
|
txData={txData}
|
||||||
|
pubKey={pubKey}
|
||||||
|
blockData={blockData}
|
||||||
|
/>
|
||||||
|
<TableRow modifier="bordered">
|
||||||
|
<TableCell>{isTeam ? t('Team ID') : t('Referral code')}</TableCell>
|
||||||
|
<TableCell>{id}</TableCell>
|
||||||
|
</TableRow>
|
||||||
|
</TableWithTbody>
|
||||||
|
|
||||||
|
<ReferralTeam
|
||||||
|
tx={txData.command.createReferralSet}
|
||||||
|
id={id}
|
||||||
|
creator={txData.submitter}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
@ -28,6 +28,10 @@ import { TxProposal } from './tx-proposal';
|
|||||||
import { TxDetailsTransfer } from './tx-transfer';
|
import { TxDetailsTransfer } from './tx-transfer';
|
||||||
import { TxDetailsStopOrderSubmission } from './tx-stop-order-submission';
|
import { TxDetailsStopOrderSubmission } from './tx-stop-order-submission';
|
||||||
import { TxDetailsLiquiditySubmission } from './tx-liquidity-submission';
|
import { TxDetailsLiquiditySubmission } from './tx-liquidity-submission';
|
||||||
|
import { TxDetailsCreateReferralSet } from './tx-create-referral-set';
|
||||||
|
import { TxDetailsApplyReferralCode } from './tx-apply-referral-code';
|
||||||
|
import { TxDetailsUpdateReferralSet } from './tx-update-referral-set';
|
||||||
|
import { TxDetailsJoinTeam } from './tx-join-team';
|
||||||
|
|
||||||
interface TxDetailsWrapperProps {
|
interface TxDetailsWrapperProps {
|
||||||
txData: BlockExplorerTransactionResult | undefined;
|
txData: BlockExplorerTransactionResult | undefined;
|
||||||
@ -121,6 +125,14 @@ function getTransactionComponent(txData?: BlockExplorerTransactionResult) {
|
|||||||
return TxDetailsStopOrderSubmission;
|
return TxDetailsStopOrderSubmission;
|
||||||
case 'Transfer Funds':
|
case 'Transfer Funds':
|
||||||
return TxDetailsTransfer;
|
return TxDetailsTransfer;
|
||||||
|
case 'Create Referral Set':
|
||||||
|
return TxDetailsCreateReferralSet;
|
||||||
|
case 'Update Referral Set':
|
||||||
|
return TxDetailsUpdateReferralSet;
|
||||||
|
case 'Apply Referral Code':
|
||||||
|
return TxDetailsApplyReferralCode;
|
||||||
|
case 'Join Team':
|
||||||
|
return TxDetailsJoinTeam;
|
||||||
default:
|
default:
|
||||||
return TxDetailsGeneric;
|
return TxDetailsGeneric;
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,47 @@
|
|||||||
|
import { t } from '@vegaprotocol/i18n';
|
||||||
|
import type { BlockExplorerTransactionResult } from '../../../routes/types/block-explorer-response';
|
||||||
|
import type { TendermintBlocksResponse } from '../../../routes/blocks/tendermint-blocks-response';
|
||||||
|
import { TxDetailsShared } from './shared/tx-details-shared';
|
||||||
|
import { TableCell, TableRow, TableWithTbody } from '../../table';
|
||||||
|
import { ReferralCodeOwner } from './referrals/referral-code-owner';
|
||||||
|
|
||||||
|
interface TxDetailsJoinTeamProps {
|
||||||
|
txData: BlockExplorerTransactionResult | undefined;
|
||||||
|
pubKey: string | undefined;
|
||||||
|
blockData: TendermintBlocksResponse | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The signature can be turned in to an id with txSignatureToDeterministicId but
|
||||||
|
*/
|
||||||
|
export const TxDetailsJoinTeam = ({
|
||||||
|
txData,
|
||||||
|
pubKey,
|
||||||
|
blockData,
|
||||||
|
}: TxDetailsJoinTeamProps) => {
|
||||||
|
if (!txData || !txData.command.joinTeam || !txData.signature) {
|
||||||
|
return <>{t('Awaiting Block Explorer transaction details')}</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const team = txData.command.joinTeam.id;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<TableWithTbody className="mb-8" allowWrap={true}>
|
||||||
|
<TxDetailsShared
|
||||||
|
txData={txData}
|
||||||
|
pubKey={pubKey}
|
||||||
|
blockData={blockData}
|
||||||
|
/>
|
||||||
|
<TableRow modifier="bordered">
|
||||||
|
<TableCell>{t('Team')}</TableCell>
|
||||||
|
<TableCell>{team}</TableCell>
|
||||||
|
</TableRow>
|
||||||
|
<TableRow modifier="bordered">
|
||||||
|
<TableCell>{t('Referrer')}</TableCell>
|
||||||
|
<ReferralCodeOwner code={team} />
|
||||||
|
</TableRow>
|
||||||
|
</TableWithTbody>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
@ -0,0 +1,54 @@
|
|||||||
|
import { t } from '@vegaprotocol/i18n';
|
||||||
|
import type { BlockExplorerTransactionResult } from '../../../routes/types/block-explorer-response';
|
||||||
|
import type { TendermintBlocksResponse } from '../../../routes/blocks/tendermint-blocks-response';
|
||||||
|
import { TxDetailsShared } from './shared/tx-details-shared';
|
||||||
|
import { TableCell, TableRow, TableWithTbody } from '../../table';
|
||||||
|
import { txSignatureToDeterministicId } from '../lib/deterministic-ids';
|
||||||
|
import { ReferralTeam } from './referrals/team';
|
||||||
|
|
||||||
|
interface TxDetailsUpdateReferralProps {
|
||||||
|
txData: BlockExplorerTransactionResult | undefined;
|
||||||
|
pubKey: string | undefined;
|
||||||
|
blockData: TendermintBlocksResponse | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A copy of create referral set, effectively.
|
||||||
|
* Updating a referral set without a team doesn't make sense,
|
||||||
|
* but is valid, so this component ignores sense.
|
||||||
|
*/
|
||||||
|
export const TxDetailsUpdateReferralSet = ({
|
||||||
|
txData,
|
||||||
|
pubKey,
|
||||||
|
blockData,
|
||||||
|
}: TxDetailsUpdateReferralProps) => {
|
||||||
|
if (!txData || !txData.command.updateReferralSet || !txData.signature) {
|
||||||
|
return <>{t('Awaiting Block Explorer transaction details')}</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const id = txSignatureToDeterministicId(txData.signature.value);
|
||||||
|
|
||||||
|
const isTeam = txData.command.updateReferralSet.isTeam || false;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<TableWithTbody className="mb-8" allowWrap={true}>
|
||||||
|
<TxDetailsShared
|
||||||
|
txData={txData}
|
||||||
|
pubKey={pubKey}
|
||||||
|
blockData={blockData}
|
||||||
|
/>
|
||||||
|
<TableRow modifier="bordered">
|
||||||
|
<TableCell>{isTeam ? t('Team ID') : t('Referral code')}</TableCell>
|
||||||
|
<TableCell>{id}</TableCell>
|
||||||
|
</TableRow>
|
||||||
|
</TableWithTbody>
|
||||||
|
|
||||||
|
<ReferralTeam
|
||||||
|
tx={txData.command.updateReferralSet}
|
||||||
|
id={id}
|
||||||
|
creator={txData.submitter}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
@ -28,6 +28,7 @@ export type FilterOption =
|
|||||||
| 'Delegate'
|
| 'Delegate'
|
||||||
| 'Ethereum Key Rotate Submission'
|
| 'Ethereum Key Rotate Submission'
|
||||||
| 'Issue Signatures'
|
| 'Issue Signatures'
|
||||||
|
| 'Join Team'
|
||||||
| 'Key Rotate Submission'
|
| 'Key Rotate Submission'
|
||||||
| 'Liquidity Provision Order'
|
| 'Liquidity Provision Order'
|
||||||
| 'Node Signature'
|
| 'Node Signature'
|
||||||
@ -47,30 +48,32 @@ export type FilterOption =
|
|||||||
| 'Vote on Proposal'
|
| 'Vote on Proposal'
|
||||||
| 'Withdraw';
|
| 'Withdraw';
|
||||||
|
|
||||||
// Alphabetised list of transaction types to appear at the top level
|
export const filterOptions: Record<string, FilterOption[]> = {
|
||||||
export const PrimaryFilterOptions: FilterOption[] = [
|
'Market Instructions': [
|
||||||
'Amend LiquidityProvision Order',
|
'Amend LiquidityProvision Order',
|
||||||
'Amend Order',
|
'Amend Order',
|
||||||
'Batch Market Instructions',
|
'Batch Market Instructions',
|
||||||
'Cancel LiquidityProvision Order',
|
'Cancel LiquidityProvision Order',
|
||||||
'Cancel Order',
|
'Cancel Order',
|
||||||
'Cancel Transfer Funds',
|
|
||||||
'Delegate',
|
|
||||||
'Liquidity Provision Order',
|
'Liquidity Provision Order',
|
||||||
'Proposal',
|
|
||||||
'Stop Orders Submission',
|
'Stop Orders Submission',
|
||||||
'Stop Orders Cancellation',
|
'Stop Orders Cancellation',
|
||||||
'Submit Oracle Data',
|
|
||||||
'Submit Order',
|
'Submit Order',
|
||||||
|
],
|
||||||
|
'Transfers and Withdrawals': [
|
||||||
'Transfer Funds',
|
'Transfer Funds',
|
||||||
'Undelegate',
|
'Cancel Transfer Funds',
|
||||||
'Vote on Proposal',
|
|
||||||
'Withdraw',
|
'Withdraw',
|
||||||
];
|
],
|
||||||
|
Governance: ['Delegate', 'Undelegate', 'Vote on Proposal', 'Proposal'],
|
||||||
// Alphabetised list of transaction types to nest under a 'More...' submenu
|
Referrals: [
|
||||||
export const SecondaryFilterOptions: FilterOption[] = [
|
'Apply Referral Code',
|
||||||
'Chain Event',
|
'Create Referral Set',
|
||||||
|
'Join Team',
|
||||||
|
'Update Referral Set',
|
||||||
|
],
|
||||||
|
'External Data': ['Chain Event', 'Submit Oracle Data'],
|
||||||
|
Validators: [
|
||||||
'Ethereum Key Rotate Submission',
|
'Ethereum Key Rotate Submission',
|
||||||
'Issue Signatures',
|
'Issue Signatures',
|
||||||
'Key Rotate Submission',
|
'Key Rotate Submission',
|
||||||
@ -80,12 +83,11 @@ export const SecondaryFilterOptions: FilterOption[] = [
|
|||||||
'Register new Node',
|
'Register new Node',
|
||||||
'State Variable Proposal',
|
'State Variable Proposal',
|
||||||
'Validator Heartbeat',
|
'Validator Heartbeat',
|
||||||
];
|
],
|
||||||
|
};
|
||||||
|
|
||||||
export const AllFilterOptions: FilterOption[] = [
|
export const AllFilterOptions: FilterOption[] =
|
||||||
...PrimaryFilterOptions,
|
Object.values(filterOptions).flat();
|
||||||
...SecondaryFilterOptions,
|
|
||||||
];
|
|
||||||
|
|
||||||
export interface TxFilterProps {
|
export interface TxFilterProps {
|
||||||
filters: Set<FilterOption>;
|
filters: Set<FilterOption>;
|
||||||
@ -122,29 +124,15 @@ export const TxsFilter = ({ filters, setFilters }: TxFilterProps) => {
|
|||||||
<DropdownMenuSeparator />
|
<DropdownMenuSeparator />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{PrimaryFilterOptions.map((f) => (
|
|
||||||
<DropdownMenuCheckboxItem
|
{Object.entries(filterOptions).map(([key, value]) => (
|
||||||
key={f}
|
<DropdownMenuSub key={key}>
|
||||||
checked={filters.has(f)}
|
|
||||||
onCheckedChange={() => {
|
|
||||||
// NOTE: These act like radio buttons until the API supports multiple filters
|
|
||||||
setFilters(new Set([f]));
|
|
||||||
}}
|
|
||||||
id={`radio-${f}`}
|
|
||||||
>
|
|
||||||
{f}
|
|
||||||
<DropdownMenuItemIndicator>
|
|
||||||
<Icon name="tick-circle" />
|
|
||||||
</DropdownMenuItemIndicator>
|
|
||||||
</DropdownMenuCheckboxItem>
|
|
||||||
))}
|
|
||||||
<DropdownMenuSub>
|
|
||||||
<DropdownMenuSubTrigger>
|
<DropdownMenuSubTrigger>
|
||||||
{t('More Types')}
|
{t(key)}
|
||||||
<Icon name="chevron-right" />
|
<Icon name="chevron-right" />
|
||||||
</DropdownMenuSubTrigger>
|
</DropdownMenuSubTrigger>
|
||||||
<DropdownMenuSubContent>
|
<DropdownMenuSubContent>
|
||||||
{SecondaryFilterOptions.map((f) => (
|
{value.map((f) => (
|
||||||
<DropdownMenuCheckboxItem
|
<DropdownMenuCheckboxItem
|
||||||
key={f}
|
key={f}
|
||||||
checked={filters.has(f)}
|
checked={filters.has(f)}
|
||||||
@ -162,6 +150,7 @@ export const TxsFilter = ({ filters, setFilters }: TxFilterProps) => {
|
|||||||
))}
|
))}
|
||||||
</DropdownMenuSubContent>
|
</DropdownMenuSubContent>
|
||||||
</DropdownMenuSub>
|
</DropdownMenuSub>
|
||||||
|
))}
|
||||||
</DropdownMenuContent>
|
</DropdownMenuContent>
|
||||||
</DropdownMenu>
|
</DropdownMenu>
|
||||||
);
|
);
|
||||||
|
@ -48,6 +48,8 @@ const displayString: StringMap = {
|
|||||||
StopOrdersSubmission: 'Stop',
|
StopOrdersSubmission: 'Stop',
|
||||||
StopOrdersCancellation: 'Cancel stop',
|
StopOrdersCancellation: 'Cancel stop',
|
||||||
'Stop Orders Cancellation': 'Cancel stop',
|
'Stop Orders Cancellation': 'Cancel stop',
|
||||||
|
'Apply Referral Code': 'Referral',
|
||||||
|
'Create Referral Set': 'Create referral',
|
||||||
};
|
};
|
||||||
|
|
||||||
export function getLabelForStopOrderType(
|
export function getLabelForStopOrderType(
|
||||||
|
@ -31,6 +31,10 @@ const Tx = () => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!data || !data?.transaction) {
|
||||||
|
errorMessage = 'Transaction not found';
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<section>
|
<section>
|
||||||
<PageHeader
|
<PageHeader
|
||||||
@ -49,7 +53,7 @@ const Tx = () => {
|
|||||||
<TxDetails
|
<TxDetails
|
||||||
className="mb-28"
|
className="mb-28"
|
||||||
txData={data?.transaction}
|
txData={data?.transaction}
|
||||||
pubKey={data?.transaction.submitter}
|
pubKey={data?.transaction?.submitter}
|
||||||
/>
|
/>
|
||||||
</RenderFetched>
|
</RenderFetched>
|
||||||
</section>
|
</section>
|
||||||
|
@ -11,8 +11,8 @@ interface TxDetailsProps {
|
|||||||
export const txDetailsTruncateLength = 30;
|
export const txDetailsTruncateLength = 30;
|
||||||
|
|
||||||
export const TxDetails = ({ txData, pubKey }: TxDetailsProps) => {
|
export const TxDetails = ({ txData, pubKey }: TxDetailsProps) => {
|
||||||
if (!txData) {
|
if (!txData || !pubKey) {
|
||||||
return <>{t('Awaiting Block Explorer transaction details')}</>;
|
return <>{t('Transaction could not be found')}</>;
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<section className="mb-10" key={txData.hash}>
|
<section className="mb-10" key={txData.hash}>
|
||||||
|
Loading…
Reference in New Issue
Block a user