chore(trading): team page refactor, games with specified epoch from (#5772)
This commit is contained in:
parent
c7dd5e846a
commit
0d3bcf05a1
@ -3,8 +3,8 @@ import { ErrorBoundary } from '@sentry/react';
|
|||||||
import { CompetitionsHeader } from '../../components/competitions/competitions-header';
|
import { CompetitionsHeader } from '../../components/competitions/competitions-header';
|
||||||
import { Intent, Loader, TradingButton } from '@vegaprotocol/ui-toolkit';
|
import { Intent, Loader, TradingButton } from '@vegaprotocol/ui-toolkit';
|
||||||
|
|
||||||
import { useGames } from '../../lib/hooks/use-games';
|
import { useGameCards } from '../../lib/hooks/use-game-cards';
|
||||||
import { useCurrentEpochInfoQuery } from '../referrals/hooks/__generated__/Epoch';
|
import { useCurrentEpochInfoQuery } from '../../lib/hooks/__generated__/Epoch';
|
||||||
import { Link, useNavigate } from 'react-router-dom';
|
import { Link, useNavigate } from 'react-router-dom';
|
||||||
import { Links } from '../../lib/links';
|
import { Links } from '../../lib/links';
|
||||||
import {
|
import {
|
||||||
@ -28,7 +28,7 @@ export const CompetitionsHome = () => {
|
|||||||
const { data: epochData } = useCurrentEpochInfoQuery();
|
const { data: epochData } = useCurrentEpochInfoQuery();
|
||||||
const currentEpoch = Number(epochData?.epoch.id);
|
const currentEpoch = Number(epochData?.epoch.id);
|
||||||
|
|
||||||
const { data: gamesData, loading: gamesLoading } = useGames({
|
const { data: gamesData, loading: gamesLoading } = useGameCards({
|
||||||
onlyActive: true,
|
onlyActive: true,
|
||||||
currentEpoch,
|
currentEpoch,
|
||||||
});
|
});
|
||||||
|
@ -12,7 +12,6 @@ import {
|
|||||||
type TeamStats as ITeamStats,
|
type TeamStats as ITeamStats,
|
||||||
type Team as TeamType,
|
type Team as TeamType,
|
||||||
type Member,
|
type Member,
|
||||||
type TeamGame,
|
|
||||||
} from '../../lib/hooks/use-team';
|
} from '../../lib/hooks/use-team';
|
||||||
import { DApp, EXPLORER_PARTIES, useLinks } from '@vegaprotocol/environment';
|
import { DApp, EXPLORER_PARTIES, useLinks } from '@vegaprotocol/environment';
|
||||||
import { TeamAvatar } from '../../components/competitions/team-avatar';
|
import { TeamAvatar } from '../../components/competitions/team-avatar';
|
||||||
@ -23,6 +22,11 @@ import { LayoutWithGradient } from '../../components/layouts-inner';
|
|||||||
import { useVegaWallet } from '@vegaprotocol/wallet';
|
import { useVegaWallet } from '@vegaprotocol/wallet';
|
||||||
import { JoinTeam } from './join-team';
|
import { JoinTeam } from './join-team';
|
||||||
import { UpdateTeamButton } from './update-team-button';
|
import { UpdateTeamButton } from './update-team-button';
|
||||||
|
import {
|
||||||
|
type TeamGame,
|
||||||
|
useGames,
|
||||||
|
areTeamGames,
|
||||||
|
} from '../../lib/hooks/use-games';
|
||||||
|
|
||||||
export const CompetitionsTeam = () => {
|
export const CompetitionsTeam = () => {
|
||||||
const t = useT();
|
const t = useT();
|
||||||
@ -38,8 +42,12 @@ export const CompetitionsTeam = () => {
|
|||||||
const TeamPageContainer = ({ teamId }: { teamId: string | undefined }) => {
|
const TeamPageContainer = ({ teamId }: { teamId: string | undefined }) => {
|
||||||
const t = useT();
|
const t = useT();
|
||||||
const { pubKey } = useVegaWallet();
|
const { pubKey } = useVegaWallet();
|
||||||
const { data, team, partyTeam, stats, members, games, loading, refetch } =
|
const { data, team, partyTeam, stats, members, loading, refetch } = useTeam(
|
||||||
useTeam(teamId, pubKey || undefined);
|
teamId,
|
||||||
|
pubKey || undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const { data: games, loading: gamesLoading } = useGames(teamId);
|
||||||
|
|
||||||
// only show spinner on first load so when users join teams its smoother
|
// only show spinner on first load so when users join teams its smoother
|
||||||
if (!data && loading) {
|
if (!data && loading) {
|
||||||
@ -64,7 +72,8 @@ const TeamPageContainer = ({ teamId }: { teamId: string | undefined }) => {
|
|||||||
partyTeam={partyTeam}
|
partyTeam={partyTeam}
|
||||||
stats={stats}
|
stats={stats}
|
||||||
members={members}
|
members={members}
|
||||||
games={games}
|
games={areTeamGames(games) ? games : undefined}
|
||||||
|
gamesLoading={gamesLoading}
|
||||||
refetch={refetch}
|
refetch={refetch}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
@ -76,6 +85,7 @@ const TeamPage = ({
|
|||||||
stats,
|
stats,
|
||||||
members,
|
members,
|
||||||
games,
|
games,
|
||||||
|
gamesLoading,
|
||||||
refetch,
|
refetch,
|
||||||
}: {
|
}: {
|
||||||
team: TeamType;
|
team: TeamType;
|
||||||
@ -83,6 +93,7 @@ const TeamPage = ({
|
|||||||
stats?: ITeamStats;
|
stats?: ITeamStats;
|
||||||
members?: Member[];
|
members?: Member[];
|
||||||
games?: TeamGame[];
|
games?: TeamGame[];
|
||||||
|
gamesLoading?: boolean;
|
||||||
refetch: () => void;
|
refetch: () => void;
|
||||||
}) => {
|
}) => {
|
||||||
const t = useT();
|
const t = useT();
|
||||||
@ -113,7 +124,11 @@ const TeamPage = ({
|
|||||||
onClick={() => setShowGames(true)}
|
onClick={() => setShowGames(true)}
|
||||||
data-testid="games-toggle"
|
data-testid="games-toggle"
|
||||||
>
|
>
|
||||||
{t('Games ({{count}})', { count: games ? games.length : 0 })}
|
{t('Games {{games}}', {
|
||||||
|
replace: {
|
||||||
|
games: gamesLoading ? '' : games ? `(${games.length})` : '(0)',
|
||||||
|
},
|
||||||
|
})}
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton
|
<ToggleButton
|
||||||
active={!showGames}
|
active={!showGames}
|
||||||
@ -125,15 +140,33 @@ const TeamPage = ({
|
|||||||
})}
|
})}
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</div>
|
</div>
|
||||||
{showGames ? <Games games={games} /> : <Members members={members} />}
|
{showGames ? (
|
||||||
|
<Games games={games} gamesLoading={gamesLoading} />
|
||||||
|
) : (
|
||||||
|
<Members members={members} />
|
||||||
|
)}
|
||||||
</section>
|
</section>
|
||||||
</LayoutWithGradient>
|
</LayoutWithGradient>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const Games = ({ games }: { games?: TeamGame[] }) => {
|
const Games = ({
|
||||||
|
games,
|
||||||
|
gamesLoading,
|
||||||
|
}: {
|
||||||
|
games?: TeamGame[];
|
||||||
|
gamesLoading?: boolean;
|
||||||
|
}) => {
|
||||||
const t = useT();
|
const t = useT();
|
||||||
|
|
||||||
|
if (gamesLoading) {
|
||||||
|
return (
|
||||||
|
<div className="w-[15px]">
|
||||||
|
<Loader size="small" />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (!games?.length) {
|
if (!games?.length) {
|
||||||
return <p>{t('No games')}</p>;
|
return <p>{t('No games')}</p>;
|
||||||
}
|
}
|
||||||
|
@ -11,7 +11,7 @@ import { useEffect } from 'react';
|
|||||||
import { useT } from '../../../lib/use-t';
|
import { useT } from '../../../lib/use-t';
|
||||||
import { matchPath, useLocation, useNavigate } from 'react-router-dom';
|
import { matchPath, useLocation, useNavigate } from 'react-router-dom';
|
||||||
import { Routes } from '../../../lib/links';
|
import { Routes } from '../../../lib/links';
|
||||||
import { useCurrentEpochInfoQuery } from './__generated__/Epoch';
|
import { useCurrentEpochInfoQuery } from '../../../lib/hooks/__generated__/Epoch';
|
||||||
|
|
||||||
const REFETCH_INTERVAL = 60 * 60 * 1000; // 1h
|
const REFETCH_INTERVAL = 60 * 60 * 1000; // 1h
|
||||||
const NON_ELIGIBLE_REFERRAL_SET_TOAST_ID = 'non-eligible-referral-set';
|
const NON_ELIGIBLE_REFERRAL_SET_TOAST_ID = 'non-eligible-referral-set';
|
||||||
|
@ -34,9 +34,10 @@ import {
|
|||||||
} from './hooks/use-referral';
|
} from './hooks/use-referral';
|
||||||
import { ApplyCodeForm, ApplyCodeFormContainer } from './apply-code-form';
|
import { ApplyCodeForm, ApplyCodeFormContainer } from './apply-code-form';
|
||||||
import { useReferralProgram } from './hooks/use-referral-program';
|
import { useReferralProgram } from './hooks/use-referral-program';
|
||||||
import { useCurrentEpochInfoQuery } from './hooks/__generated__/Epoch';
|
import { useCurrentEpochInfoQuery } from '../../lib/hooks/__generated__/Epoch';
|
||||||
import { QUSDTooltip } from './qusd-tooltip';
|
import { QUSDTooltip } from './qusd-tooltip';
|
||||||
import { CodeTile, StatTile, Tile } from './tile';
|
import { CodeTile, StatTile, Tile } from './tile';
|
||||||
|
import { areTeamGames, useGames } from '../../lib/hooks/use-games';
|
||||||
|
|
||||||
export const ReferralStatistics = () => {
|
export const ReferralStatistics = () => {
|
||||||
const { pubKey } = useVegaWallet();
|
const { pubKey } = useVegaWallet();
|
||||||
@ -576,7 +577,8 @@ export const RefereesTable = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const Team = ({ teamId }: { teamId?: string }) => {
|
const Team = ({ teamId }: { teamId?: string }) => {
|
||||||
const { team, games, members } = useTeam(teamId);
|
const { team, members } = useTeam(teamId);
|
||||||
|
const { data: games } = useGames(teamId);
|
||||||
|
|
||||||
if (!team) return null;
|
if (!team) return null;
|
||||||
|
|
||||||
@ -585,7 +587,10 @@ const Team = ({ teamId }: { teamId?: string }) => {
|
|||||||
<TeamAvatar teamId={team.teamId} imgUrl={team.avatarUrl} />
|
<TeamAvatar teamId={team.teamId} imgUrl={team.avatarUrl} />
|
||||||
<div className="flex flex-col items-start gap-1 lg:gap-3">
|
<div className="flex flex-col items-start gap-1 lg:gap-3">
|
||||||
<h1 className="calt text-2xl lg:text-3xl xl:text-5xl">{team.name}</h1>
|
<h1 className="calt text-2xl lg:text-3xl xl:text-5xl">{team.name}</h1>
|
||||||
<TeamStats members={members} games={games} />
|
<TeamStats
|
||||||
|
members={members}
|
||||||
|
games={areTeamGames(games) ? games : undefined}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</Tile>
|
</Tile>
|
||||||
);
|
);
|
||||||
|
@ -153,5 +153,8 @@ const cacheConfig: InMemoryCacheConfig = {
|
|||||||
OrderUpdate: {
|
OrderUpdate: {
|
||||||
keyFields: false,
|
keyFields: false,
|
||||||
},
|
},
|
||||||
|
Game: {
|
||||||
|
keyFields: false,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { type TeamGame, type TeamStats } from '../../lib/hooks/use-team';
|
import { type TeamStats } from '../../lib/hooks/use-team';
|
||||||
import { type TeamsFieldsFragment } from '../../lib/hooks/__generated__/Teams';
|
import { type TeamsFieldsFragment } from '../../lib/hooks/__generated__/Teams';
|
||||||
import { TeamAvatar, getFallbackAvatar } from './team-avatar';
|
import { TeamAvatar, getFallbackAvatar } from './team-avatar';
|
||||||
import { FavoriteGame, Stat } from './team-stats';
|
import { FavoriteGame, Stat } from './team-stats';
|
||||||
@ -13,6 +13,7 @@ import { take } from 'lodash';
|
|||||||
import { DispatchMetricLabels } from '@vegaprotocol/types';
|
import { DispatchMetricLabels } from '@vegaprotocol/types';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import { UpdateTeamButton } from '../../client-pages/competitions/update-team-button';
|
import { UpdateTeamButton } from '../../client-pages/competitions/update-team-button';
|
||||||
|
import { type TeamGame } from '../../lib/hooks/use-games';
|
||||||
|
|
||||||
export const TeamCard = ({
|
export const TeamCard = ({
|
||||||
rank,
|
rank,
|
||||||
|
@ -11,11 +11,11 @@ import { formatNumberRounded } from '@vegaprotocol/utils';
|
|||||||
import {
|
import {
|
||||||
type TeamStats as ITeamStats,
|
type TeamStats as ITeamStats,
|
||||||
type Member,
|
type Member,
|
||||||
type TeamGame,
|
|
||||||
} from '../../lib/hooks/use-team';
|
} from '../../lib/hooks/use-team';
|
||||||
import { useT } from '../../lib/use-t';
|
import { useT } from '../../lib/use-t';
|
||||||
import { DispatchMetricLabels, type DispatchMetric } from '@vegaprotocol/types';
|
import { DispatchMetricLabels, type DispatchMetric } from '@vegaprotocol/types';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
|
import { type TeamGame } from '../../lib/hooks/use-games';
|
||||||
|
|
||||||
export const TeamStats = ({
|
export const TeamStats = ({
|
||||||
stats,
|
stats,
|
||||||
|
@ -213,12 +213,12 @@ def create_team(vega: VegaServiceNull):
|
|||||||
|
|
||||||
def test_team_page_games_table(team_page: Page):
|
def test_team_page_games_table(team_page: Page):
|
||||||
team_page.get_by_test_id("games-toggle").click()
|
team_page.get_by_test_id("games-toggle").click()
|
||||||
expect(team_page.get_by_test_id("games-toggle")).to_have_text("Games (1)")
|
expect(team_page.get_by_test_id("games-toggle")).to_have_text("Games (10)")
|
||||||
expect(team_page.get_by_test_id("rank-0")).to_have_text("2")
|
expect(team_page.get_by_test_id("rank-0")).to_have_text("2")
|
||||||
expect(team_page.get_by_test_id("epoch-0")).to_have_text("19")
|
expect(team_page.get_by_test_id("epoch-0")).to_have_text("19")
|
||||||
expect(team_page.get_by_test_id("type-0")
|
expect(team_page.get_by_test_id("type-0")
|
||||||
).to_have_text("Price maker fees paid")
|
).to_have_text("Price maker fees paid")
|
||||||
expect(team_page.get_by_test_id("amount-0")).to_have_text("74")
|
expect(team_page.get_by_test_id("amount-0")).to_have_text("74") # 7,438,330 on preview.11
|
||||||
expect(team_page.get_by_test_id("participatingTeams-0")).to_have_text("2")
|
expect(team_page.get_by_test_id("participatingTeams-0")).to_have_text("2")
|
||||||
expect(team_page.get_by_test_id("participatingMembers-0")).to_have_text("4")
|
expect(team_page.get_by_test_id("participatingMembers-0")).to_have_text("4")
|
||||||
|
|
||||||
|
35
apps/trading/lib/hooks/Games.graphql
Normal file
35
apps/trading/lib/hooks/Games.graphql
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
fragment TeamEntity on TeamGameEntity {
|
||||||
|
rank
|
||||||
|
volume
|
||||||
|
rewardMetric
|
||||||
|
rewardEarned
|
||||||
|
totalRewardsEarned
|
||||||
|
team {
|
||||||
|
teamId
|
||||||
|
membersParticipating {
|
||||||
|
individual
|
||||||
|
rank
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fragment GameFields on Game {
|
||||||
|
id
|
||||||
|
epoch
|
||||||
|
numberOfParticipants
|
||||||
|
entities {
|
||||||
|
... on TeamGameEntity {
|
||||||
|
...TeamEntity
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
query Games($epochFrom: Int) {
|
||||||
|
games(epochFrom: $epochFrom, entityScope: ENTITY_SCOPE_TEAMS) {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
...GameFields
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -29,28 +29,6 @@ fragment TeamRefereeFields on TeamReferee {
|
|||||||
joinedAtEpoch
|
joinedAtEpoch
|
||||||
}
|
}
|
||||||
|
|
||||||
fragment TeamEntity on TeamGameEntity {
|
|
||||||
rank
|
|
||||||
volume
|
|
||||||
rewardMetric
|
|
||||||
rewardEarned
|
|
||||||
totalRewardsEarned
|
|
||||||
team {
|
|
||||||
teamId
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fragment TeamGameFields on Game {
|
|
||||||
id
|
|
||||||
epoch
|
|
||||||
numberOfParticipants
|
|
||||||
entities {
|
|
||||||
... on TeamGameEntity {
|
|
||||||
...TeamEntity
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fragment TeamMemberStatsFields on TeamMemberStatistics {
|
fragment TeamMemberStatsFields on TeamMemberStatistics {
|
||||||
partyId
|
partyId
|
||||||
totalQuantumVolume
|
totalQuantumVolume
|
||||||
@ -87,13 +65,6 @@ query Team($teamId: ID!, $partyId: ID, $aggregationEpochs: Int) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
games(entityScope: ENTITY_SCOPE_TEAMS) {
|
|
||||||
edges {
|
|
||||||
node {
|
|
||||||
...TeamGameFields
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
teamMembersStatistics(
|
teamMembersStatistics(
|
||||||
teamId: $teamId
|
teamId: $teamId
|
||||||
aggregationEpochs: $aggregationEpochs
|
aggregationEpochs: $aggregationEpochs
|
||||||
|
83
apps/trading/lib/hooks/__generated__/Games.ts
generated
Normal file
83
apps/trading/lib/hooks/__generated__/Games.ts
generated
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
import * as Types from '@vegaprotocol/types';
|
||||||
|
|
||||||
|
import { gql } from '@apollo/client';
|
||||||
|
import * as Apollo from '@apollo/client';
|
||||||
|
const defaultOptions = {} as const;
|
||||||
|
export type TeamEntityFragment = { __typename?: 'TeamGameEntity', rank: number, volume: string, rewardMetric: Types.DispatchMetric, rewardEarned: string, totalRewardsEarned: string, team: { __typename?: 'TeamParticipation', teamId: string, membersParticipating: Array<{ __typename?: 'IndividualGameEntity', individual: string, rank: number }> } };
|
||||||
|
|
||||||
|
export type GameFieldsFragment = { __typename?: 'Game', id: string, epoch: number, numberOfParticipants: number, entities: Array<{ __typename?: 'IndividualGameEntity' } | { __typename?: 'TeamGameEntity', rank: number, volume: string, rewardMetric: Types.DispatchMetric, rewardEarned: string, totalRewardsEarned: string, team: { __typename?: 'TeamParticipation', teamId: string, membersParticipating: Array<{ __typename?: 'IndividualGameEntity', individual: string, rank: number }> } }> };
|
||||||
|
|
||||||
|
export type GamesQueryVariables = Types.Exact<{
|
||||||
|
epochFrom?: Types.InputMaybe<Types.Scalars['Int']>;
|
||||||
|
}>;
|
||||||
|
|
||||||
|
|
||||||
|
export type GamesQuery = { __typename?: 'Query', games: { __typename?: 'GamesConnection', edges?: Array<{ __typename?: 'GameEdge', node: { __typename?: 'Game', id: string, epoch: number, numberOfParticipants: number, entities: Array<{ __typename?: 'IndividualGameEntity' } | { __typename?: 'TeamGameEntity', rank: number, volume: string, rewardMetric: Types.DispatchMetric, rewardEarned: string, totalRewardsEarned: string, team: { __typename?: 'TeamParticipation', teamId: string, membersParticipating: Array<{ __typename?: 'IndividualGameEntity', individual: string, rank: number }> } }> } } | null> | null } };
|
||||||
|
|
||||||
|
export const TeamEntityFragmentDoc = gql`
|
||||||
|
fragment TeamEntity on TeamGameEntity {
|
||||||
|
rank
|
||||||
|
volume
|
||||||
|
rewardMetric
|
||||||
|
rewardEarned
|
||||||
|
totalRewardsEarned
|
||||||
|
team {
|
||||||
|
teamId
|
||||||
|
membersParticipating {
|
||||||
|
individual
|
||||||
|
rank
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
export const GameFieldsFragmentDoc = gql`
|
||||||
|
fragment GameFields on Game {
|
||||||
|
id
|
||||||
|
epoch
|
||||||
|
numberOfParticipants
|
||||||
|
entities {
|
||||||
|
... on TeamGameEntity {
|
||||||
|
...TeamEntity
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
${TeamEntityFragmentDoc}`;
|
||||||
|
export const GamesDocument = gql`
|
||||||
|
query Games($epochFrom: Int) {
|
||||||
|
games(epochFrom: $epochFrom, entityScope: ENTITY_SCOPE_TEAMS) {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
...GameFields
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
${GameFieldsFragmentDoc}`;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* __useGamesQuery__
|
||||||
|
*
|
||||||
|
* To run a query within a React component, call `useGamesQuery` and pass it any options that fit your needs.
|
||||||
|
* When your component renders, `useGamesQuery` 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 } = useGamesQuery({
|
||||||
|
* variables: {
|
||||||
|
* epochFrom: // value for 'epochFrom'
|
||||||
|
* },
|
||||||
|
* });
|
||||||
|
*/
|
||||||
|
export function useGamesQuery(baseOptions?: Apollo.QueryHookOptions<GamesQuery, GamesQueryVariables>) {
|
||||||
|
const options = {...defaultOptions, ...baseOptions}
|
||||||
|
return Apollo.useQuery<GamesQuery, GamesQueryVariables>(GamesDocument, options);
|
||||||
|
}
|
||||||
|
export function useGamesLazyQuery(baseOptions?: Apollo.LazyQueryHookOptions<GamesQuery, GamesQueryVariables>) {
|
||||||
|
const options = {...defaultOptions, ...baseOptions}
|
||||||
|
return Apollo.useLazyQuery<GamesQuery, GamesQueryVariables>(GamesDocument, options);
|
||||||
|
}
|
||||||
|
export type GamesQueryHookResult = ReturnType<typeof useGamesQuery>;
|
||||||
|
export type GamesLazyQueryHookResult = ReturnType<typeof useGamesLazyQuery>;
|
||||||
|
export type GamesQueryResult = Apollo.QueryResult<GamesQuery, GamesQueryVariables>;
|
38
apps/trading/lib/hooks/__generated__/Team.ts
generated
38
apps/trading/lib/hooks/__generated__/Team.ts
generated
@ -9,10 +9,6 @@ export type TeamStatsFieldsFragment = { __typename?: 'TeamStatistics', teamId: s
|
|||||||
|
|
||||||
export type TeamRefereeFieldsFragment = { __typename?: 'TeamReferee', teamId: string, referee: string, joinedAt: any, joinedAtEpoch: number };
|
export type TeamRefereeFieldsFragment = { __typename?: 'TeamReferee', teamId: string, referee: string, joinedAt: any, joinedAtEpoch: number };
|
||||||
|
|
||||||
export type TeamEntityFragment = { __typename?: 'TeamGameEntity', rank: number, volume: string, rewardMetric: Types.DispatchMetric, rewardEarned: string, totalRewardsEarned: string, team: { __typename?: 'TeamParticipation', teamId: string } };
|
|
||||||
|
|
||||||
export type TeamGameFieldsFragment = { __typename?: 'Game', id: string, epoch: number, numberOfParticipants: number, entities: Array<{ __typename?: 'IndividualGameEntity' } | { __typename?: 'TeamGameEntity', rank: number, volume: string, rewardMetric: Types.DispatchMetric, rewardEarned: string, totalRewardsEarned: string, team: { __typename?: 'TeamParticipation', teamId: string } }> };
|
|
||||||
|
|
||||||
export type TeamMemberStatsFieldsFragment = { __typename?: 'TeamMemberStatistics', partyId: string, totalQuantumVolume: string, totalQuantumRewards: string, totalGamesPlayed: number };
|
export type TeamMemberStatsFieldsFragment = { __typename?: 'TeamMemberStatistics', partyId: string, totalQuantumVolume: string, totalQuantumRewards: string, totalGamesPlayed: number };
|
||||||
|
|
||||||
export type TeamQueryVariables = Types.Exact<{
|
export type TeamQueryVariables = Types.Exact<{
|
||||||
@ -22,7 +18,7 @@ export type TeamQueryVariables = Types.Exact<{
|
|||||||
}>;
|
}>;
|
||||||
|
|
||||||
|
|
||||||
export type TeamQuery = { __typename?: 'Query', teams?: { __typename?: 'TeamConnection', edges: Array<{ __typename?: 'TeamEdge', node: { __typename?: 'Team', teamId: string, referrer: string, name: string, teamUrl: string, avatarUrl: string, createdAt: any, createdAtEpoch: number, closed: boolean, allowList: Array<string> } }> } | null, partyTeams?: { __typename?: 'TeamConnection', edges: Array<{ __typename?: 'TeamEdge', node: { __typename?: 'Team', teamId: string, referrer: string, name: string, teamUrl: string, avatarUrl: string, createdAt: any, createdAtEpoch: number, closed: boolean, allowList: Array<string> } }> } | null, teamsStatistics?: { __typename?: 'TeamsStatisticsConnection', edges: Array<{ __typename?: 'TeamStatisticsEdge', node: { __typename?: 'TeamStatistics', teamId: string, totalQuantumVolume: string, totalQuantumRewards: string, totalGamesPlayed: number, gamesPlayed: Array<string>, quantumRewards: Array<{ __typename?: 'QuantumRewardsPerEpoch', epoch: number, totalQuantumRewards: string }> } }> } | null, teamReferees?: { __typename?: 'TeamRefereeConnection', edges: Array<{ __typename?: 'TeamRefereeEdge', node: { __typename?: 'TeamReferee', teamId: string, referee: string, joinedAt: any, joinedAtEpoch: number } }> } | null, games: { __typename?: 'GamesConnection', edges?: Array<{ __typename?: 'GameEdge', node: { __typename?: 'Game', id: string, epoch: number, numberOfParticipants: number, entities: Array<{ __typename?: 'IndividualGameEntity' } | { __typename?: 'TeamGameEntity', rank: number, volume: string, rewardMetric: Types.DispatchMetric, rewardEarned: string, totalRewardsEarned: string, team: { __typename?: 'TeamParticipation', teamId: string } }> } } | null> | null }, teamMembersStatistics?: { __typename?: 'TeamMembersStatisticsConnection', edges: Array<{ __typename?: 'TeamMemberStatisticsEdge', node: { __typename?: 'TeamMemberStatistics', partyId: string, totalQuantumVolume: string, totalQuantumRewards: string, totalGamesPlayed: number } }> } | null };
|
export type TeamQuery = { __typename?: 'Query', teams?: { __typename?: 'TeamConnection', edges: Array<{ __typename?: 'TeamEdge', node: { __typename?: 'Team', teamId: string, referrer: string, name: string, teamUrl: string, avatarUrl: string, createdAt: any, createdAtEpoch: number, closed: boolean, allowList: Array<string> } }> } | null, partyTeams?: { __typename?: 'TeamConnection', edges: Array<{ __typename?: 'TeamEdge', node: { __typename?: 'Team', teamId: string, referrer: string, name: string, teamUrl: string, avatarUrl: string, createdAt: any, createdAtEpoch: number, closed: boolean, allowList: Array<string> } }> } | null, teamsStatistics?: { __typename?: 'TeamsStatisticsConnection', edges: Array<{ __typename?: 'TeamStatisticsEdge', node: { __typename?: 'TeamStatistics', teamId: string, totalQuantumVolume: string, totalQuantumRewards: string, totalGamesPlayed: number, gamesPlayed: Array<string>, quantumRewards: Array<{ __typename?: 'QuantumRewardsPerEpoch', epoch: number, totalQuantumRewards: string }> } }> } | null, teamReferees?: { __typename?: 'TeamRefereeConnection', edges: Array<{ __typename?: 'TeamRefereeEdge', node: { __typename?: 'TeamReferee', teamId: string, referee: string, joinedAt: any, joinedAtEpoch: number } }> } | null, teamMembersStatistics?: { __typename?: 'TeamMembersStatisticsConnection', edges: Array<{ __typename?: 'TeamMemberStatisticsEdge', node: { __typename?: 'TeamMemberStatistics', partyId: string, totalQuantumVolume: string, totalQuantumRewards: string, totalGamesPlayed: number } }> } | null };
|
||||||
|
|
||||||
export const TeamFieldsFragmentDoc = gql`
|
export const TeamFieldsFragmentDoc = gql`
|
||||||
fragment TeamFields on Team {
|
fragment TeamFields on Team {
|
||||||
@ -58,30 +54,6 @@ export const TeamRefereeFieldsFragmentDoc = gql`
|
|||||||
joinedAtEpoch
|
joinedAtEpoch
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
export const TeamEntityFragmentDoc = gql`
|
|
||||||
fragment TeamEntity on TeamGameEntity {
|
|
||||||
rank
|
|
||||||
volume
|
|
||||||
rewardMetric
|
|
||||||
rewardEarned
|
|
||||||
totalRewardsEarned
|
|
||||||
team {
|
|
||||||
teamId
|
|
||||||
}
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
export const TeamGameFieldsFragmentDoc = gql`
|
|
||||||
fragment TeamGameFields on Game {
|
|
||||||
id
|
|
||||||
epoch
|
|
||||||
numberOfParticipants
|
|
||||||
entities {
|
|
||||||
... on TeamGameEntity {
|
|
||||||
...TeamEntity
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
${TeamEntityFragmentDoc}`;
|
|
||||||
export const TeamMemberStatsFieldsFragmentDoc = gql`
|
export const TeamMemberStatsFieldsFragmentDoc = gql`
|
||||||
fragment TeamMemberStatsFields on TeamMemberStatistics {
|
fragment TeamMemberStatsFields on TeamMemberStatistics {
|
||||||
partyId
|
partyId
|
||||||
@ -120,13 +92,6 @@ export const TeamDocument = gql`
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
games(entityScope: ENTITY_SCOPE_TEAMS) {
|
|
||||||
edges {
|
|
||||||
node {
|
|
||||||
...TeamGameFields
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
teamMembersStatistics(teamId: $teamId, aggregationEpochs: $aggregationEpochs) {
|
teamMembersStatistics(teamId: $teamId, aggregationEpochs: $aggregationEpochs) {
|
||||||
edges {
|
edges {
|
||||||
node {
|
node {
|
||||||
@ -138,7 +103,6 @@ export const TeamDocument = gql`
|
|||||||
${TeamFieldsFragmentDoc}
|
${TeamFieldsFragmentDoc}
|
||||||
${TeamStatsFieldsFragmentDoc}
|
${TeamStatsFieldsFragmentDoc}
|
||||||
${TeamRefereeFieldsFragmentDoc}
|
${TeamRefereeFieldsFragmentDoc}
|
||||||
${TeamGameFieldsFragmentDoc}
|
|
||||||
${TeamMemberStatsFieldsFragmentDoc}`;
|
${TeamMemberStatsFieldsFragmentDoc}`;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
48
apps/trading/lib/hooks/use-game-cards.ts
Normal file
48
apps/trading/lib/hooks/use-game-cards.ts
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
import compact from 'lodash/compact';
|
||||||
|
import { useActiveRewardsQuery } from '../../components/rewards-container/__generated__/Rewards';
|
||||||
|
import { isActiveReward } from '../../components/rewards-container/active-rewards';
|
||||||
|
import {
|
||||||
|
EntityScope,
|
||||||
|
IndividualScope,
|
||||||
|
type TransferNode,
|
||||||
|
} from '@vegaprotocol/types';
|
||||||
|
|
||||||
|
const isScopedToTeams = (node: TransferNode) =>
|
||||||
|
node.transfer.kind.__typename === 'RecurringTransfer' &&
|
||||||
|
// scoped to teams
|
||||||
|
(node.transfer.kind.dispatchStrategy?.entityScope ===
|
||||||
|
EntityScope.ENTITY_SCOPE_TEAMS ||
|
||||||
|
// or to individuals
|
||||||
|
(node.transfer.kind.dispatchStrategy?.entityScope ===
|
||||||
|
EntityScope.ENTITY_SCOPE_INDIVIDUALS &&
|
||||||
|
// but they have to be in a team
|
||||||
|
node.transfer.kind.dispatchStrategy.individualScope ===
|
||||||
|
IndividualScope.INDIVIDUAL_SCOPE_IN_TEAM));
|
||||||
|
|
||||||
|
export const useGameCards = ({
|
||||||
|
currentEpoch,
|
||||||
|
onlyActive,
|
||||||
|
}: {
|
||||||
|
currentEpoch: number;
|
||||||
|
onlyActive: boolean;
|
||||||
|
}) => {
|
||||||
|
const { data, loading, error } = useActiveRewardsQuery({
|
||||||
|
variables: {
|
||||||
|
isReward: true,
|
||||||
|
},
|
||||||
|
fetchPolicy: 'cache-and-network',
|
||||||
|
});
|
||||||
|
|
||||||
|
const games = compact(data?.transfersConnection?.edges?.map((n) => n?.node))
|
||||||
|
.map((n) => n as TransferNode)
|
||||||
|
.filter((node) => {
|
||||||
|
const active = onlyActive ? isActiveReward(node, currentEpoch) : true;
|
||||||
|
return active && isScopedToTeams(node);
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: games,
|
||||||
|
loading,
|
||||||
|
error,
|
||||||
|
};
|
||||||
|
};
|
@ -1,48 +1,80 @@
|
|||||||
import compact from 'lodash/compact';
|
|
||||||
import { useActiveRewardsQuery } from '../../components/rewards-container/__generated__/Rewards';
|
|
||||||
import { isActiveReward } from '../../components/rewards-container/active-rewards';
|
|
||||||
import {
|
import {
|
||||||
EntityScope,
|
useGamesQuery,
|
||||||
IndividualScope,
|
type GameFieldsFragment,
|
||||||
type TransferNode,
|
type TeamEntityFragment,
|
||||||
} from '@vegaprotocol/types';
|
} from './__generated__/Games';
|
||||||
|
import orderBy from 'lodash/orderBy';
|
||||||
|
import { removePaginationWrapper } from '@vegaprotocol/utils';
|
||||||
|
import { useCurrentEpochInfoQuery } from './__generated__/Epoch';
|
||||||
|
import { type ApolloError } from '@apollo/client';
|
||||||
|
|
||||||
const isScopedToTeams = (node: TransferNode) =>
|
const TAKE_EPOCHS = 30; // TODO: should this be DEFAULT_AGGREGATION_EPOCHS?
|
||||||
node.transfer.kind.__typename === 'RecurringTransfer' &&
|
|
||||||
// scoped to teams
|
|
||||||
(node.transfer.kind.dispatchStrategy?.entityScope ===
|
|
||||||
EntityScope.ENTITY_SCOPE_TEAMS ||
|
|
||||||
// or to individuals
|
|
||||||
(node.transfer.kind.dispatchStrategy?.entityScope ===
|
|
||||||
EntityScope.ENTITY_SCOPE_INDIVIDUALS &&
|
|
||||||
// but they have to be in a team
|
|
||||||
node.transfer.kind.dispatchStrategy.individualScope ===
|
|
||||||
IndividualScope.INDIVIDUAL_SCOPE_IN_TEAM));
|
|
||||||
|
|
||||||
export const useGames = ({
|
const findTeam = (entities: GameFieldsFragment['entities'], teamId: string) => {
|
||||||
currentEpoch,
|
const team = entities.find(
|
||||||
onlyActive,
|
(ent) => ent.__typename === 'TeamGameEntity' && ent.team.teamId === teamId
|
||||||
}: {
|
);
|
||||||
currentEpoch: number;
|
if (team?.__typename === 'TeamGameEntity') return team; // drops __typename === 'IndividualGameEntity' from team object
|
||||||
onlyActive: boolean;
|
return undefined;
|
||||||
}) => {
|
};
|
||||||
const { data, loading, error } = useActiveRewardsQuery({
|
|
||||||
|
export type Game = GameFieldsFragment & {
|
||||||
|
/** The team entity data accessible only if scoped to particular team. */
|
||||||
|
team?: TeamEntityFragment;
|
||||||
|
};
|
||||||
|
export type TeamGame = Game & { team: NonNullable<Game['team']> };
|
||||||
|
|
||||||
|
const isTeamGame = (game: Game): game is TeamGame => game.team !== undefined;
|
||||||
|
export const areTeamGames = (games?: Game[]): games is TeamGame[] =>
|
||||||
|
Boolean(games && games.filter((g) => isTeamGame(g)).length > 0);
|
||||||
|
|
||||||
|
type GamesData = {
|
||||||
|
data?: Game[];
|
||||||
|
loading: boolean;
|
||||||
|
error?: ApolloError;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useGames = (teamId?: string, epochFrom?: number): GamesData => {
|
||||||
|
const {
|
||||||
|
data: epochData,
|
||||||
|
loading: epochLoading,
|
||||||
|
error: epochError,
|
||||||
|
} = useCurrentEpochInfoQuery({
|
||||||
|
skip: Boolean(epochFrom),
|
||||||
|
});
|
||||||
|
|
||||||
|
let from = epochFrom;
|
||||||
|
if (!from && epochData) {
|
||||||
|
from = Number(epochData.epoch.id) - TAKE_EPOCHS;
|
||||||
|
if (from < 1) from = 1; // make sure it's not negative
|
||||||
|
}
|
||||||
|
|
||||||
|
const { data, loading, error } = useGamesQuery({
|
||||||
variables: {
|
variables: {
|
||||||
isReward: true,
|
epochFrom: from,
|
||||||
},
|
},
|
||||||
|
skip: !from,
|
||||||
fetchPolicy: 'cache-and-network',
|
fetchPolicy: 'cache-and-network',
|
||||||
|
context: { isEnlargedTimeout: true },
|
||||||
});
|
});
|
||||||
|
|
||||||
const games = compact(data?.transfersConnection?.edges?.map((n) => n?.node))
|
const allGames = removePaginationWrapper(data?.games.edges);
|
||||||
.map((n) => n as TransferNode)
|
const allOrScoped = allGames
|
||||||
.filter((node) => {
|
.map((g) => ({
|
||||||
const active = onlyActive ? isActiveReward(node, currentEpoch) : true;
|
...g,
|
||||||
return active && isScopedToTeams(node);
|
team: teamId ? findTeam(g.entities, teamId) : undefined,
|
||||||
|
}))
|
||||||
|
.filter((g) => {
|
||||||
|
// passthrough if not scoped to particular team
|
||||||
|
if (!teamId) return true;
|
||||||
|
return isTeamGame(g);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const games = orderBy(allOrScoped, 'epoch', 'desc');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
data: games,
|
data: games,
|
||||||
loading,
|
loading: loading || epochLoading,
|
||||||
error,
|
error: error || epochError,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -4,6 +4,7 @@ import first from 'lodash/first';
|
|||||||
import { useTeamsQuery } from './__generated__/Teams';
|
import { useTeamsQuery } from './__generated__/Teams';
|
||||||
import { useTeam } from './use-team';
|
import { useTeam } from './use-team';
|
||||||
import { useTeams } from './use-teams';
|
import { useTeams } from './use-teams';
|
||||||
|
import { areTeamGames, useGames } from './use-games';
|
||||||
|
|
||||||
export const useMyTeam = () => {
|
export const useMyTeam = () => {
|
||||||
const { pubKey } = useVegaWallet();
|
const { pubKey } = useVegaWallet();
|
||||||
@ -19,7 +20,8 @@ export const useMyTeam = () => {
|
|||||||
|
|
||||||
const team = first(compact(maybeMyTeam?.teams?.edges.map((n) => n.node)));
|
const team = first(compact(maybeMyTeam?.teams?.edges.map((n) => n.node)));
|
||||||
const rank = teams.findIndex((t) => t.teamId === team?.teamId) + 1;
|
const rank = teams.findIndex((t) => t.teamId === team?.teamId) + 1;
|
||||||
const { games, stats } = useTeam(team?.teamId);
|
const { stats } = useTeam(team?.teamId);
|
||||||
|
const { data: games } = useGames(team?.teamId);
|
||||||
|
|
||||||
return { team, stats, games, rank };
|
return { team, stats, games: areTeamGames(games) ? games : undefined, rank };
|
||||||
};
|
};
|
||||||
|
@ -1,11 +1,8 @@
|
|||||||
import compact from 'lodash/compact';
|
|
||||||
import orderBy from 'lodash/orderBy';
|
|
||||||
import {
|
import {
|
||||||
useTeamQuery,
|
useTeamQuery,
|
||||||
type TeamFieldsFragment,
|
type TeamFieldsFragment,
|
||||||
type TeamStatsFieldsFragment,
|
type TeamStatsFieldsFragment,
|
||||||
type TeamRefereeFieldsFragment,
|
type TeamRefereeFieldsFragment,
|
||||||
type TeamEntityFragment,
|
|
||||||
type TeamMemberStatsFieldsFragment,
|
type TeamMemberStatsFieldsFragment,
|
||||||
} from './__generated__/Team';
|
} from './__generated__/Team';
|
||||||
import { DEFAULT_AGGREGATION_EPOCHS } from './use-teams';
|
import { DEFAULT_AGGREGATION_EPOCHS } from './use-teams';
|
||||||
@ -18,8 +15,6 @@ export type Member = TeamRefereeFieldsFragment & {
|
|||||||
totalQuantumVolume: string;
|
totalQuantumVolume: string;
|
||||||
totalQuantumRewards: string;
|
totalQuantumRewards: string;
|
||||||
};
|
};
|
||||||
export type TeamEntity = TeamEntityFragment;
|
|
||||||
export type TeamGame = ReturnType<typeof useTeam>['games'][number];
|
|
||||||
export type MemberStats = TeamMemberStatsFieldsFragment;
|
export type MemberStats = TeamMemberStatsFieldsFragment;
|
||||||
|
|
||||||
export const useTeam = (teamId?: string, partyId?: string) => {
|
export const useTeam = (teamId?: string, partyId?: string) => {
|
||||||
@ -80,33 +75,11 @@ export const useTeam = (teamId?: string, partyId?: string) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find games where the current team participated in
|
|
||||||
const gamesWithTeam = compact(data?.games.edges).map((edge) => {
|
|
||||||
const team = edge.node.entities.find((e) => {
|
|
||||||
if (e.__typename !== 'TeamGameEntity') return false;
|
|
||||||
if (e.team.teamId !== teamId) return false;
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!team) return null;
|
|
||||||
|
|
||||||
return {
|
|
||||||
id: edge.node.id,
|
|
||||||
epoch: edge.node.epoch,
|
|
||||||
numberOfParticipants: edge.node.numberOfParticipants,
|
|
||||||
entities: edge.node.entities,
|
|
||||||
team: team as TeamEntity, // TS can't infer that all the game entities are teams
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const games = orderBy(compact(gamesWithTeam), 'epoch', 'desc');
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...queryResult,
|
...queryResult,
|
||||||
stats: teamStatsEdge?.node,
|
stats: teamStatsEdge?.node,
|
||||||
team,
|
team,
|
||||||
members,
|
members,
|
||||||
games,
|
|
||||||
partyTeam,
|
partyTeam,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -48,5 +48,8 @@ export const DEFAULT_CACHE_CONFIG: InMemoryCacheConfig = {
|
|||||||
statistics: {
|
statistics: {
|
||||||
keyFields: false,
|
keyFields: false,
|
||||||
},
|
},
|
||||||
|
Game: {
|
||||||
|
keyFields: false,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user