2022-10-06 21:41:22 +02:00
|
|
|
/* eslint-disable jest/require-top-level-describe */
|
|
|
|
import React from 'react';
|
|
|
|
import { fireEvent, render } from '@testing-library/react';
|
2022-12-06 20:51:48 +01:00
|
|
|
import { renderWithUserEvent } from '../../../../test/lib/render-helpers';
|
|
|
|
|
2022-10-06 21:41:22 +02:00
|
|
|
import { SIZES } from '../../../helpers/constants/design-system';
|
|
|
|
|
2022-11-15 19:06:59 +01:00
|
|
|
import Box from '../../ui/box';
|
|
|
|
|
2022-10-06 21:41:22 +02:00
|
|
|
import { TextFieldBase } from './text-field-base';
|
|
|
|
|
|
|
|
describe('TextFieldBase', () => {
|
|
|
|
it('should render correctly', () => {
|
2022-12-06 20:51:48 +01:00
|
|
|
const { getByRole, container } = render(<TextFieldBase />);
|
2022-10-06 21:41:22 +02:00
|
|
|
expect(getByRole('textbox')).toBeDefined();
|
2022-12-06 20:51:48 +01:00
|
|
|
expect(container).toMatchSnapshot();
|
2022-10-06 21:41:22 +02:00
|
|
|
});
|
|
|
|
it('should render and be able to input text', () => {
|
|
|
|
const { getByTestId } = render(
|
|
|
|
<TextFieldBase inputProps={{ 'data-testid': 'text-field-base' }} />,
|
|
|
|
);
|
|
|
|
const textFieldBase = getByTestId('text-field-base');
|
|
|
|
|
|
|
|
expect(textFieldBase.value).toBe(''); // initial value is empty string
|
|
|
|
fireEvent.change(textFieldBase, { target: { value: 'text value' } });
|
|
|
|
expect(textFieldBase.value).toBe('text value');
|
|
|
|
fireEvent.change(textFieldBase, { target: { value: '' } }); // reset value
|
|
|
|
expect(textFieldBase.value).toBe(''); // value is empty string after reset
|
|
|
|
});
|
2022-12-06 20:51:48 +01:00
|
|
|
it('should render with focused state when clicked', async () => {
|
|
|
|
const { getByTestId, user } = renderWithUserEvent(
|
2022-10-26 00:23:48 +02:00
|
|
|
<TextFieldBase
|
|
|
|
data-testid="text-field-base"
|
|
|
|
inputProps={{ 'data-testid': 'input' }}
|
|
|
|
/>,
|
|
|
|
);
|
2022-12-06 20:51:48 +01:00
|
|
|
const textFieldBase = getByTestId('input');
|
2022-10-26 00:23:48 +02:00
|
|
|
|
2022-12-06 20:51:48 +01:00
|
|
|
await user.click(textFieldBase);
|
2022-10-26 00:23:48 +02:00
|
|
|
expect(getByTestId('input')).toHaveFocus();
|
|
|
|
expect(getByTestId('text-field-base')).toHaveClass(
|
|
|
|
'mm-text-field-base--focused ',
|
|
|
|
);
|
|
|
|
});
|
2022-12-06 20:51:48 +01:00
|
|
|
it('should render and fire onFocus and onBlur events', async () => {
|
2022-10-06 21:41:22 +02:00
|
|
|
const onFocus = jest.fn();
|
|
|
|
const onBlur = jest.fn();
|
2022-12-06 20:51:48 +01:00
|
|
|
const { getByTestId, user } = renderWithUserEvent(
|
2022-10-06 21:41:22 +02:00
|
|
|
<TextFieldBase
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base' }}
|
|
|
|
onFocus={onFocus}
|
|
|
|
onBlur={onBlur}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
|
2022-12-06 20:51:48 +01:00
|
|
|
const textFieldBase = getByTestId('text-field-base');
|
|
|
|
await user.click(textFieldBase);
|
2022-10-06 21:41:22 +02:00
|
|
|
expect(onFocus).toHaveBeenCalledTimes(1);
|
|
|
|
fireEvent.blur(textFieldBase);
|
|
|
|
expect(onBlur).toHaveBeenCalledTimes(1);
|
|
|
|
});
|
2022-12-06 20:51:48 +01:00
|
|
|
it('should render and fire onChange event', async () => {
|
2022-10-06 21:41:22 +02:00
|
|
|
const onChange = jest.fn();
|
2022-12-06 20:51:48 +01:00
|
|
|
const { getByTestId, user } = renderWithUserEvent(
|
2022-10-06 21:41:22 +02:00
|
|
|
<TextFieldBase
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base' }}
|
|
|
|
onChange={onChange}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
const textFieldBase = getByTestId('text-field-base');
|
2022-12-06 20:51:48 +01:00
|
|
|
await user.type(textFieldBase, '123');
|
|
|
|
expect(textFieldBase).toHaveValue('123');
|
|
|
|
expect(onChange).toHaveBeenCalledTimes(3);
|
2022-10-06 21:41:22 +02:00
|
|
|
});
|
2022-12-06 20:51:48 +01:00
|
|
|
it('should render and fire onClick event', async () => {
|
2022-10-06 21:41:22 +02:00
|
|
|
const onClick = jest.fn();
|
2022-12-06 20:51:48 +01:00
|
|
|
const { getByTestId, user } = renderWithUserEvent(
|
2022-10-06 21:41:22 +02:00
|
|
|
<TextFieldBase
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base' }}
|
|
|
|
onClick={onClick}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
const textFieldBase = getByTestId('text-field-base');
|
|
|
|
|
2022-12-06 20:51:48 +01:00
|
|
|
await user.click(textFieldBase);
|
2022-10-06 21:41:22 +02:00
|
|
|
expect(onClick).toHaveBeenCalledTimes(1);
|
|
|
|
});
|
|
|
|
it('should render with different size classes', () => {
|
|
|
|
const { getByTestId } = render(
|
|
|
|
<>
|
|
|
|
<TextFieldBase size={SIZES.SM} data-testid="sm" />
|
|
|
|
<TextFieldBase size={SIZES.MD} data-testid="md" />
|
|
|
|
<TextFieldBase size={SIZES.LG} data-testid="lg" />
|
|
|
|
</>,
|
|
|
|
);
|
|
|
|
expect(getByTestId('sm')).toHaveClass('mm-text-field-base--size-sm');
|
|
|
|
expect(getByTestId('md')).toHaveClass('mm-text-field-base--size-md');
|
|
|
|
expect(getByTestId('lg')).toHaveClass('mm-text-field-base--size-lg');
|
|
|
|
});
|
|
|
|
it('should render with different types', () => {
|
|
|
|
const { getByTestId } = render(
|
|
|
|
<>
|
|
|
|
<TextFieldBase inputProps={{ 'data-testid': 'text-field-base-text' }} />
|
|
|
|
<TextFieldBase
|
|
|
|
type="number"
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base-number' }}
|
|
|
|
/>
|
|
|
|
<TextFieldBase
|
|
|
|
type="password"
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base-password' }}
|
|
|
|
/>
|
|
|
|
</>,
|
|
|
|
);
|
|
|
|
expect(getByTestId('text-field-base-text')).toHaveAttribute('type', 'text');
|
|
|
|
expect(getByTestId('text-field-base-number')).toHaveAttribute(
|
|
|
|
'type',
|
|
|
|
'number',
|
|
|
|
);
|
|
|
|
expect(getByTestId('text-field-base-password')).toHaveAttribute(
|
|
|
|
'type',
|
|
|
|
'password',
|
|
|
|
);
|
|
|
|
});
|
2022-10-26 00:23:48 +02:00
|
|
|
it('should render with truncate class as true by default and remove it when truncate is false', () => {
|
2022-10-06 21:41:22 +02:00
|
|
|
const { getByTestId } = render(
|
2022-10-26 00:23:48 +02:00
|
|
|
<>
|
|
|
|
<TextFieldBase data-testid="truncate" />
|
|
|
|
<TextFieldBase truncate={false} data-testid="no-truncate" />
|
|
|
|
</>,
|
2022-10-06 21:41:22 +02:00
|
|
|
);
|
|
|
|
expect(getByTestId('truncate')).toHaveClass('mm-text-field-base--truncate');
|
2022-10-26 00:23:48 +02:00
|
|
|
expect(getByTestId('no-truncate')).not.toHaveClass(
|
|
|
|
'mm-text-field-base--truncate',
|
|
|
|
);
|
2022-10-06 21:41:22 +02:00
|
|
|
});
|
|
|
|
it('should render with right and left accessories', () => {
|
|
|
|
const { getByRole, getByText } = render(
|
|
|
|
<TextFieldBase
|
|
|
|
leftAccessory={<div>left accessory</div>}
|
|
|
|
rightAccessory={<div>right accessory</div>}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
expect(getByRole('textbox')).toBeDefined();
|
|
|
|
expect(getByText('left accessory')).toBeDefined();
|
|
|
|
expect(getByText('right accessory')).toBeDefined();
|
|
|
|
});
|
|
|
|
it('should render with working ref using inputRef prop', () => {
|
|
|
|
// Because the 'ref' attribute wont flow down to the DOM
|
|
|
|
// I'm not exactly sure how to test this?
|
|
|
|
const mockRef = jest.fn();
|
|
|
|
const { getByRole } = render(<TextFieldBase inputRef={mockRef} />);
|
|
|
|
expect(getByRole('textbox')).toBeDefined();
|
|
|
|
expect(mockRef).toHaveBeenCalledTimes(1);
|
|
|
|
});
|
|
|
|
it('should render with autoComplete', () => {
|
|
|
|
const { getByTestId } = render(
|
|
|
|
<TextFieldBase
|
|
|
|
autoComplete
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base-auto-complete' }}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
expect(getByTestId('text-field-base-auto-complete')).toHaveAttribute(
|
|
|
|
'autocomplete',
|
|
|
|
'on',
|
|
|
|
);
|
|
|
|
});
|
|
|
|
it('should render with autoFocus', () => {
|
|
|
|
const { getByRole } = render(<TextFieldBase autoFocus />);
|
|
|
|
expect(getByRole('textbox')).toHaveFocus();
|
|
|
|
});
|
|
|
|
it('should render with a defaultValue', () => {
|
|
|
|
const { getByRole } = render(
|
|
|
|
<TextFieldBase
|
|
|
|
defaultValue="default value"
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base-default-value' }}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
expect(getByRole('textbox').value).toBe('default value');
|
|
|
|
});
|
2022-12-06 20:51:48 +01:00
|
|
|
it('should render in disabled state and not focus or be clickable', async () => {
|
2022-10-06 21:41:22 +02:00
|
|
|
const mockOnClick = jest.fn();
|
|
|
|
const mockOnFocus = jest.fn();
|
2022-12-06 20:51:48 +01:00
|
|
|
const { getByRole, getByTestId, user } = renderWithUserEvent(
|
|
|
|
<TextFieldBase
|
|
|
|
disabled
|
|
|
|
onFocus={mockOnFocus}
|
|
|
|
onClick={mockOnClick}
|
|
|
|
data-testid="text-field-base"
|
|
|
|
/>,
|
2022-10-06 21:41:22 +02:00
|
|
|
);
|
|
|
|
|
2022-12-06 20:51:48 +01:00
|
|
|
const textFieldBase = getByTestId('text-field-base');
|
|
|
|
|
|
|
|
await user.click(textFieldBase);
|
2022-10-06 21:41:22 +02:00
|
|
|
expect(getByRole('textbox')).toBeDisabled();
|
|
|
|
expect(mockOnClick).toHaveBeenCalledTimes(0);
|
|
|
|
expect(mockOnFocus).toHaveBeenCalledTimes(0);
|
|
|
|
});
|
|
|
|
it('should render with error className when error is true', () => {
|
|
|
|
const { getByTestId } = render(
|
2022-11-15 19:06:59 +01:00
|
|
|
<TextFieldBase error data-testid="text-field-base-error" />,
|
2022-10-06 21:41:22 +02:00
|
|
|
);
|
|
|
|
expect(getByTestId('text-field-base-error')).toHaveClass(
|
|
|
|
'mm-text-field-base--error',
|
|
|
|
);
|
|
|
|
});
|
|
|
|
it('should render with maxLength and not allow more than the set characters', async () => {
|
2022-12-06 20:51:48 +01:00
|
|
|
const { getByRole, user } = renderWithUserEvent(
|
|
|
|
<TextFieldBase maxLength={5} />,
|
|
|
|
);
|
2022-10-06 21:41:22 +02:00
|
|
|
const textFieldBase = getByRole('textbox');
|
2022-12-06 20:51:48 +01:00
|
|
|
await user.type(textFieldBase, '1234567890');
|
2022-10-06 21:41:22 +02:00
|
|
|
expect(getByRole('textbox')).toBeDefined();
|
|
|
|
expect(textFieldBase.maxLength).toBe(5);
|
|
|
|
expect(textFieldBase.value).toBe('12345');
|
|
|
|
expect(textFieldBase.value).toHaveLength(5);
|
|
|
|
});
|
2022-12-06 20:51:48 +01:00
|
|
|
it('should render with readOnly attr when readOnly is true', async () => {
|
|
|
|
const { getByTestId, getByRole, user } = renderWithUserEvent(
|
|
|
|
<TextFieldBase readOnly data-testid="read-only" />,
|
2022-10-06 21:41:22 +02:00
|
|
|
);
|
2022-12-06 20:51:48 +01:00
|
|
|
const textFieldBase = getByTestId('read-only');
|
|
|
|
await user.type(textFieldBase, '1234567890');
|
|
|
|
expect(getByRole('textbox').value).toBe('');
|
|
|
|
expect(getByRole('textbox')).toHaveAttribute('readonly', '');
|
2022-10-06 21:41:22 +02:00
|
|
|
});
|
|
|
|
it('should render with required attr when required is true', () => {
|
|
|
|
const { getByTestId } = render(
|
|
|
|
<TextFieldBase
|
|
|
|
required
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base-required' }}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
expect(getByTestId('text-field-base-required')).toHaveAttribute(
|
|
|
|
'required',
|
|
|
|
'',
|
|
|
|
);
|
|
|
|
});
|
2022-12-06 20:51:48 +01:00
|
|
|
it('should render with a custom input and still work', async () => {
|
2022-11-15 19:06:59 +01:00
|
|
|
const CustomInputComponent = React.forwardRef((props, ref) => (
|
|
|
|
<Box ref={ref} as="input" {...props} />
|
|
|
|
));
|
|
|
|
CustomInputComponent.displayName = 'CustomInputComponent'; // fixes eslint error
|
2022-12-06 20:51:48 +01:00
|
|
|
const { getByTestId, user } = renderWithUserEvent(
|
2022-11-10 01:33:20 +01:00
|
|
|
<TextFieldBase
|
|
|
|
InputComponent={CustomInputComponent}
|
|
|
|
inputProps={{ 'data-testid': 'text-field-base', className: 'test' }}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
const textFieldBase = getByTestId('text-field-base');
|
|
|
|
|
|
|
|
expect(textFieldBase.value).toBe(''); // initial value is empty string
|
2022-12-06 20:51:48 +01:00
|
|
|
await user.type(textFieldBase, 'text value');
|
2022-11-10 01:33:20 +01:00
|
|
|
expect(textFieldBase.value).toBe('text value');
|
|
|
|
fireEvent.change(textFieldBase, { target: { value: '' } }); // reset value
|
|
|
|
expect(textFieldBase.value).toBe(''); // value is empty string after reset
|
|
|
|
});
|
2022-10-06 21:41:22 +02:00
|
|
|
});
|