2024-03-01 14:25:56 +00:00
|
|
|
import { act, render, screen, within } from '@testing-library/react';
|
2023-07-31 16:08:55 +00:00
|
|
|
import userEvent from '@testing-library/user-event';
|
2023-05-15 08:08:04 +00:00
|
|
|
import { MemoryRouter } from 'react-router-dom';
|
|
|
|
import { Navbar } from './navbar';
|
2023-07-31 16:08:55 +00:00
|
|
|
import { useGlobalStore } from '../../stores';
|
2023-12-12 12:53:20 +00:00
|
|
|
import { ENV, useFeatureFlags } from '@vegaprotocol/environment';
|
2024-03-01 14:25:56 +00:00
|
|
|
import {
|
|
|
|
mockConfig,
|
|
|
|
MockedWalletProvider,
|
|
|
|
} from '@vegaprotocol/wallet-react/testing';
|
2024-03-08 17:29:41 +00:00
|
|
|
import { MockedProvider, type MockedResponse } from '@apollo/react-testing';
|
|
|
|
import {
|
|
|
|
PartyProfilesDocument,
|
|
|
|
type PartyProfilesQuery,
|
|
|
|
type PartyProfilesQueryVariables,
|
|
|
|
} from '../vega-wallet-connect-button/__generated__/PartyProfiles';
|
2023-07-31 16:08:55 +00:00
|
|
|
|
|
|
|
jest.mock('@vegaprotocol/proposals', () => ({
|
|
|
|
ProtocolUpgradeCountdown: () => null,
|
|
|
|
}));
|
2023-05-15 08:08:04 +00:00
|
|
|
|
|
|
|
describe('Navbar', () => {
|
2024-03-01 14:25:56 +00:00
|
|
|
const mockKeys = [
|
2023-07-31 16:08:55 +00:00
|
|
|
{
|
2024-03-01 14:25:56 +00:00
|
|
|
name: 'Key 1',
|
|
|
|
publicKey: '1'.repeat(64),
|
2023-07-31 16:08:55 +00:00
|
|
|
},
|
|
|
|
{
|
2024-03-01 14:25:56 +00:00
|
|
|
name: 'Key 2',
|
|
|
|
publicKey: '2'.repeat(64),
|
2023-07-31 16:08:55 +00:00
|
|
|
},
|
|
|
|
];
|
2024-03-08 17:29:41 +00:00
|
|
|
const key1Alias = 'key 1 alias';
|
2023-07-31 16:08:55 +00:00
|
|
|
const marketId = 'abc';
|
|
|
|
const navbarContent = 'navbar-menu-content';
|
|
|
|
|
2024-03-08 17:29:41 +00:00
|
|
|
const partyProfilesMock: MockedResponse<
|
|
|
|
PartyProfilesQuery,
|
|
|
|
PartyProfilesQueryVariables
|
|
|
|
> = {
|
|
|
|
request: {
|
|
|
|
query: PartyProfilesDocument,
|
|
|
|
variables: {
|
|
|
|
partyIds: mockKeys.map((k) => k.publicKey),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: {
|
|
|
|
partiesProfilesConnection: {
|
|
|
|
edges: [
|
|
|
|
{
|
|
|
|
node: {
|
|
|
|
partyId: mockKeys[0].publicKey,
|
|
|
|
alias: key1Alias,
|
|
|
|
metadata: [],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
const renderComponent = (initialEntries?: string[]) => {
|
2023-07-31 16:08:55 +00:00
|
|
|
return render(
|
|
|
|
<MemoryRouter initialEntries={initialEntries}>
|
2024-03-08 17:29:41 +00:00
|
|
|
<MockedProvider mocks={[partyProfilesMock]}>
|
|
|
|
<MockedWalletProvider>
|
|
|
|
<Navbar />
|
|
|
|
</MockedWalletProvider>
|
|
|
|
</MockedProvider>
|
2023-07-31 16:08:55 +00:00
|
|
|
</MemoryRouter>
|
2023-05-15 08:08:04 +00:00
|
|
|
);
|
2023-07-31 16:08:55 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
useGlobalStore.setState({ marketId });
|
2023-12-12 12:53:20 +00:00
|
|
|
useFeatureFlags.setState({ flags: { REFERRALS: true } });
|
2023-09-21 13:25:19 +00:00
|
|
|
const mockedENV = jest.mocked(ENV);
|
|
|
|
mockedENV.VEGA_TOKEN_URL = 'governance';
|
|
|
|
});
|
|
|
|
|
|
|
|
afterAll(() => {
|
|
|
|
jest.clearAllMocks();
|
2023-07-31 16:08:55 +00:00
|
|
|
});
|
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
afterEach(() => {
|
|
|
|
act(() => {
|
|
|
|
mockConfig.reset();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2023-07-31 16:08:55 +00:00
|
|
|
it('should be properly rendered', () => {
|
|
|
|
renderComponent();
|
|
|
|
|
|
|
|
const expectedLinks = [
|
|
|
|
['/', ''],
|
2023-10-16 18:18:26 +00:00
|
|
|
['/markets', 'Markets'],
|
2023-07-31 16:08:55 +00:00
|
|
|
[`/markets/${marketId}`, 'Trading'],
|
|
|
|
['/portfolio', 'Portfolio'],
|
2023-09-21 13:25:19 +00:00
|
|
|
['/referrals', 'Referrals'],
|
2023-10-25 21:59:30 +00:00
|
|
|
['/fees', 'Fees'],
|
2023-12-05 10:19:42 +00:00
|
|
|
['/rewards', 'Rewards'],
|
2023-09-13 21:25:31 +00:00
|
|
|
[expect.stringContaining('governance'), 'Governance'],
|
2023-07-31 16:08:55 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
const links = screen.getAllByRole('link');
|
|
|
|
|
|
|
|
links.forEach((link, i) => {
|
|
|
|
const [href, text] = expectedLinks[i];
|
|
|
|
expect(link).toHaveAttribute('href', href);
|
|
|
|
expect(link).toHaveTextContent(text);
|
|
|
|
});
|
2023-05-15 08:08:04 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Markets page route should not match empty market page', () => {
|
2023-10-16 18:18:26 +00:00
|
|
|
renderComponent(['/markets']);
|
2023-07-31 16:08:55 +00:00
|
|
|
expect(screen.getByRole('link', { name: 'Markets' })).toHaveClass('active');
|
|
|
|
expect(screen.getByRole('link', { name: 'Trading' })).not.toHaveClass(
|
|
|
|
'active'
|
2023-05-15 08:08:04 +00:00
|
|
|
);
|
2023-07-31 16:08:55 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('can open menu and navigate on small screens', async () => {
|
|
|
|
renderComponent();
|
|
|
|
await userEvent.click(screen.getByRole('button', { name: 'Menu' }));
|
|
|
|
|
|
|
|
const menuEl = screen.getByTestId(navbarContent);
|
|
|
|
expect(menuEl).toBeInTheDocument();
|
|
|
|
const menu = within(menuEl);
|
|
|
|
|
|
|
|
const expectedLinks = [
|
2023-10-16 18:18:26 +00:00
|
|
|
['/markets', 'Markets'],
|
2023-07-31 16:08:55 +00:00
|
|
|
[`/markets/${marketId}`, 'Trading'],
|
|
|
|
['/portfolio', 'Portfolio'],
|
2023-09-21 13:25:19 +00:00
|
|
|
['/referrals', 'Referrals'],
|
2023-10-25 21:59:30 +00:00
|
|
|
['/fees', 'Fees'],
|
2023-12-05 10:19:42 +00:00
|
|
|
['/rewards', 'Rewards'],
|
2023-09-13 21:25:31 +00:00
|
|
|
[expect.stringContaining('governance'), 'Governance'],
|
2023-07-31 16:08:55 +00:00
|
|
|
];
|
|
|
|
const links = menu.getAllByRole('link');
|
|
|
|
links.forEach((link, i) => {
|
|
|
|
const [href, text] = expectedLinks[i];
|
|
|
|
expect(link).toHaveAttribute('href', href);
|
|
|
|
expect(link).toHaveTextContent(text);
|
|
|
|
});
|
|
|
|
|
|
|
|
await userEvent.click(screen.getByRole('button', { name: 'Close menu' }));
|
|
|
|
expect(screen.queryByTestId(navbarContent)).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can close menu by clicking overlay', async () => {
|
|
|
|
renderComponent();
|
|
|
|
await userEvent.click(screen.getByRole('button', { name: 'Menu' }));
|
|
|
|
expect(screen.getByTestId(navbarContent)).toBeInTheDocument();
|
|
|
|
await userEvent.click(screen.getByTestId('navbar-menu-overlay'));
|
|
|
|
expect(screen.queryByTestId(navbarContent)).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can open wallet menu on small screens and change pubkey', async () => {
|
2024-03-01 14:25:56 +00:00
|
|
|
mockConfig.store.setState({
|
|
|
|
status: 'connected',
|
|
|
|
keys: mockKeys,
|
|
|
|
pubKey: mockKeys[0].publicKey,
|
|
|
|
});
|
|
|
|
const mockSelectPubKey = jest.spyOn(mockConfig.store, 'setState');
|
|
|
|
renderComponent(undefined);
|
2023-07-31 16:08:55 +00:00
|
|
|
await userEvent.click(screen.getByRole('button', { name: 'Wallet' }));
|
|
|
|
|
|
|
|
const menuEl = screen.getByTestId(navbarContent);
|
|
|
|
expect(menuEl).toBeInTheDocument();
|
|
|
|
const menu = within(menuEl);
|
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
expect(menu.getAllByTestId(/key-\d+-mobile/)).toHaveLength(mockKeys.length);
|
2023-07-31 16:08:55 +00:00
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
const activeKey = within(menu.getByTestId(/key-1+-mobile/));
|
|
|
|
expect(activeKey.getByText(mockKeys[0].name)).toBeInTheDocument();
|
2023-07-31 16:08:55 +00:00
|
|
|
expect(activeKey.getByTestId('icon-tick')).toBeInTheDocument();
|
2024-03-08 17:29:41 +00:00
|
|
|
expect(screen.getByText(key1Alias)).toBeInTheDocument();
|
2023-07-31 16:08:55 +00:00
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
const inactiveKey = within(menu.getByTestId(/key-2+-mobile/));
|
|
|
|
await userEvent.click(inactiveKey.getByText(mockKeys[1].name));
|
|
|
|
expect(mockSelectPubKey).toHaveBeenCalledWith({
|
|
|
|
pubKey: mockKeys[1].publicKey,
|
|
|
|
});
|
2023-07-31 16:08:55 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('can transfer and close menu', async () => {
|
2024-03-01 14:25:56 +00:00
|
|
|
mockConfig.store.setState({
|
|
|
|
status: 'connected',
|
|
|
|
keys: mockKeys,
|
|
|
|
pubKey: mockKeys[0].publicKey,
|
|
|
|
});
|
2023-07-31 16:08:55 +00:00
|
|
|
renderComponent();
|
|
|
|
await userEvent.click(screen.getByRole('button', { name: 'Wallet' }));
|
|
|
|
|
|
|
|
const menuEl = screen.getByTestId(navbarContent);
|
|
|
|
expect(menuEl).toBeInTheDocument();
|
|
|
|
const menu = within(menuEl);
|
|
|
|
|
|
|
|
await userEvent.click(menu.getByText('Transfer'));
|
|
|
|
|
|
|
|
expect(screen.queryByTestId(navbarContent)).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('can disconnect and close menu', async () => {
|
2024-03-01 14:25:56 +00:00
|
|
|
mockConfig.store.setState({
|
|
|
|
status: 'connected',
|
|
|
|
keys: mockKeys,
|
|
|
|
pubKey: mockKeys[0].publicKey,
|
|
|
|
});
|
|
|
|
const mockDisconnect = jest.spyOn(mockConfig, 'disconnect');
|
|
|
|
renderComponent(undefined);
|
2023-07-31 16:08:55 +00:00
|
|
|
await userEvent.click(screen.getByRole('button', { name: 'Wallet' }));
|
|
|
|
|
|
|
|
const menuEl = screen.getByTestId(navbarContent);
|
|
|
|
expect(menuEl).toBeInTheDocument();
|
|
|
|
const menu = within(menuEl);
|
|
|
|
|
|
|
|
await userEvent.click(menu.getByText('Disconnect'));
|
|
|
|
|
|
|
|
expect(mockDisconnect).toHaveBeenCalled();
|
|
|
|
expect(screen.queryByTestId(navbarContent)).not.toBeInTheDocument();
|
2023-05-15 08:08:04 +00:00
|
|
|
});
|
2023-11-29 14:16:17 +00:00
|
|
|
|
|
|
|
it('does not render the language selector until we have more languages', () => {
|
|
|
|
renderComponent();
|
|
|
|
expect(screen.queryByTestId('icon-globe')).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('renders the theme switcher', async () => {
|
|
|
|
renderComponent();
|
|
|
|
await userEvent.click(screen.getByTestId('icon-moon'));
|
|
|
|
expect(screen.queryByTestId('icon-moon')).not.toBeInTheDocument();
|
|
|
|
expect(screen.getByTestId('icon-sun')).toBeInTheDocument();
|
|
|
|
|
|
|
|
await userEvent.click(screen.getByTestId('icon-sun'));
|
|
|
|
expect(screen.queryByTestId('icon-sun')).not.toBeInTheDocument();
|
|
|
|
expect(screen.getByTestId('icon-moon')).toBeInTheDocument();
|
|
|
|
});
|
2023-05-15 08:08:04 +00:00
|
|
|
});
|