mirror of
https://github.com/lbr77/SideImpactor.git
synced 2026-06-20 12:19:33 -04:00
70 lines
3.1 KiB
TypeScript
70 lines
3.1 KiB
TypeScript
import { fireEvent, render, screen } from '@testing-library/react';
|
|
import userEvent from '@testing-library/user-event';
|
|
import { describe, expect, it, vi } from 'vitest';
|
|
import type { TwoFactorContext } from '../apple-signing';
|
|
import { TwoFactorModal } from './TwoFactorModal';
|
|
|
|
function createTwoFactorContext(overrides: Partial<TwoFactorContext> = {}): TwoFactorContext {
|
|
return {
|
|
submitDeviceCode: vi.fn(),
|
|
trustedPhoneNumbers: [],
|
|
requestSms: vi.fn(),
|
|
submitSmsCode: vi.fn(),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
describe('TwoFactorModal', () => {
|
|
it('is hidden when closed', () => {
|
|
const { container } = render(<TwoFactorModal open={false} ctx={null} onCancel={() => {}} />);
|
|
expect(container.querySelector('.modal')).not.toHaveClass('open');
|
|
});
|
|
|
|
it('focuses the input when opened', async () => {
|
|
render(<TwoFactorModal open ctx={createTwoFactorContext()} onCancel={() => {}} />);
|
|
// useEffect runs a setTimeout(0) before focusing — let it tick.
|
|
await new Promise((resolve) => setTimeout(resolve, 10));
|
|
expect(screen.getByLabelText('Verification Code')).toHaveFocus();
|
|
});
|
|
|
|
it('shows an inline error when submitting an empty code', async () => {
|
|
const ctx = createTwoFactorContext();
|
|
render(<TwoFactorModal open ctx={ctx} onCancel={() => {}} />);
|
|
await userEvent.click(screen.getByRole('button', { name: 'Verify' }));
|
|
expect(screen.getByText('Please enter the verification code.')).toBeInTheDocument();
|
|
expect(ctx.submitDeviceCode).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('clears the error once the user starts typing', async () => {
|
|
render(<TwoFactorModal open ctx={createTwoFactorContext()} onCancel={() => {}} />);
|
|
await userEvent.click(screen.getByRole('button', { name: 'Verify' }));
|
|
expect(screen.getByText('Please enter the verification code.')).toBeInTheDocument();
|
|
await userEvent.type(screen.getByLabelText('Verification Code'), '1');
|
|
expect(screen.queryByText('Please enter the verification code.')).not.toBeInTheDocument();
|
|
});
|
|
|
|
it('submits a trimmed code on Enter', async () => {
|
|
const ctx = createTwoFactorContext();
|
|
render(<TwoFactorModal open ctx={ctx} onCancel={() => {}} />);
|
|
const input = screen.getByLabelText('Verification Code');
|
|
await userEvent.type(input, ' 123456 ');
|
|
fireEvent.keyDown(input, { key: 'Enter' });
|
|
expect(ctx.submitDeviceCode).toHaveBeenLastCalledWith('123456');
|
|
});
|
|
|
|
it('submits via the Verify button', async () => {
|
|
const ctx = createTwoFactorContext();
|
|
render(<TwoFactorModal open ctx={ctx} onCancel={() => {}} />);
|
|
await userEvent.type(screen.getByLabelText('Verification Code'), '654321');
|
|
await userEvent.click(screen.getByRole('button', { name: 'Verify' }));
|
|
expect(ctx.submitDeviceCode).toHaveBeenLastCalledWith('654321');
|
|
});
|
|
|
|
it('calls onCancel when Cancel is clicked', async () => {
|
|
const onCancel = vi.fn();
|
|
render(<TwoFactorModal open ctx={createTwoFactorContext()} onCancel={onCancel} />);
|
|
await userEvent.click(screen.getByRole('button', { name: 'Cancel' }));
|
|
expect(onCancel).toHaveBeenCalledTimes(1);
|
|
});
|
|
});
|