import React, { Component } from 'react'; import { mount } from 'enzyme'; import scrollIntoView from 'scroll-into-view-if-needed'; import Form from '..'; import Input from '../../input'; import Button from '../../button'; import mountTest from '../../../tests/shared/mountTest'; import rtlTest from '../../../tests/shared/rtlTest'; jest.mock('scroll-into-view-if-needed'); const delay = (timeout = 0) => new Promise(resolve => { setTimeout(resolve, timeout); }); describe('Form', () => { mountTest(Form); mountTest(Form.Item); rtlTest(Form); rtlTest(Form.Item); scrollIntoView.mockImplementation(() => {}); const errorSpy = jest.spyOn(console, 'error').mockImplementation(() => {}); async function change(wrapper, index, value) { wrapper.find(Input).at(index).simulate('change', { target: { value } }); await delay(50); wrapper.update(); } beforeEach(() => { jest.useRealTimers(); scrollIntoView.mockReset(); }); afterEach(() => { errorSpy.mockReset(); }); afterAll(() => { errorSpy.mockRestore(); scrollIntoView.mockRestore(); }); describe('List', () => { function testList(name, renderField) { it(name, async () => { const wrapper = mount(
{(fields, { add, remove }) => ( <> {fields.map(field => renderField(field))} )}
, ); async function operate(className) { wrapper.find(className).last().simulate('click'); await delay(); wrapper.update(); } await operate('.add'); expect(wrapper.find(Input).length).toBe(1); await operate('.add'); expect(wrapper.find(Input).length).toBe(2); await change(wrapper, 1, ''); wrapper.update(); expect(wrapper.find('.ant-form-item-explain').length).toBe(1); await operate('.remove'); wrapper.update(); expect(wrapper.find(Input).length).toBe(1); expect(wrapper.find('.ant-form-item-explain').length).toBe(0); }); } testList('operation correctly', field => ( )); testList('nest noStyle', field => ( )); it('correct onFinish values', async () => { async function click(wrapper, className) { wrapper.find(className).last().simulate('click'); await delay(); wrapper.update(); } const onFinish = jest.fn().mockImplementation(() => {}); const wrapper = mount(
{ if (typeof v.list[0] === 'object') { /* old version led to SyntheticEvent be passed as an value here that led to weird infinite loop somewhere and OutOfMemory crash */ v = new Error('We expect value to be a primitive here'); } onFinish(v); }} > {(fields, { add, remove }) => ( <> {fields.map(field => ( // key is in a field // eslint-disable-next-line react/jsx-key ))} )}
, ); await click(wrapper, '.add'); await change(wrapper, 0, 'input1'); wrapper.find('form').simulate('submit'); await delay(); expect(onFinish).toHaveBeenLastCalledWith({ list: ['input1'] }); await click(wrapper, '.add'); await change(wrapper, 1, 'input2'); await click(wrapper, '.add'); await change(wrapper, 2, 'input3'); wrapper.find('form').simulate('submit'); await delay(); expect(onFinish).toHaveBeenLastCalledWith({ list: ['input1', 'input2', 'input3'] }); await click(wrapper, '.remove'); // will remove first input wrapper.find('form').simulate('submit'); await delay(); expect(onFinish).toHaveBeenLastCalledWith({ list: ['input2', 'input3'] }); }); }); it('noStyle Form.Item', async () => { const onChange = jest.fn(); const wrapper = mount(
, ); await change(wrapper, 0, ''); expect(wrapper.find('.ant-form-item-explain').length).toBe(1); expect(onChange).toHaveBeenCalled(); }); it('`shouldUpdate` should work with render props', () => { mount(
{() => null}
, ); expect(errorSpy).toHaveBeenCalledWith( 'Warning: [antd: Form.Item] `children` of render props only work with `shouldUpdate`.', ); }); it('`name` should not work with render props', () => { mount(
{() => null}
, ); expect(errorSpy).toHaveBeenCalledWith( "Warning: [antd: Form.Item] Do not use `name` with `children` of render props since it's not a field.", ); }); it('children is array has name props', () => { mount(
one
two
, ); expect(errorSpy).toHaveBeenCalledWith( 'Warning: [antd: Form.Item] `children` is array of render props cannot have `name`.', ); }); describe('scrollToField', () => { function test(name, genForm) { it(name, () => { let callGetForm; const Demo = () => { const { props, getForm } = genForm(); callGetForm = getForm; return (
); }; const wrapper = mount(, { attachTo: document.body }); expect(scrollIntoView).not.toHaveBeenCalled(); const form = callGetForm(); form.scrollToField('test', { block: 'start', }); const inputNode = document.getElementById('scroll_test'); expect(scrollIntoView).toHaveBeenCalledWith(inputNode, { block: 'start', scrollMode: 'if-needed', }); wrapper.unmount(); }); } // hooks test('useForm', () => { const [form] = Form.useForm(); return { props: { form }, getForm: () => form, }; }); // ref test('ref', () => { let form; return { props: { ref: instance => { form = instance; }, }, getForm: () => form, }; }); }); it('scrollToFirstError', async () => { const onFinishFailed = jest.fn(); const wrapper = mount(
, { attachTo: document.body }, ); expect(scrollIntoView).not.toHaveBeenCalled(); wrapper.find('form').simulate('submit'); await delay(50); expect(scrollIntoView).toHaveBeenCalled(); expect(onFinishFailed).toHaveBeenCalled(); wrapper.unmount(); }); it('Form.Item should support data-*、aria-* and custom attribute', () => { const wrapper = mount(
, ); expect(wrapper.render()).toMatchSnapshot(); }); it('warning when use `name` but children is not validate element', () => { mount(
text
, ); expect(errorSpy).toHaveBeenCalledWith( 'Warning: [antd: Form.Item] `name` is only used for validate React element. If you are using Form.Item as layout display, please remove `name` instead.', ); }); it('dynamic change required', () => { const wrapper = mount(
({ required: getFieldValue('light') })]} >
, ); expect(wrapper.find('.ant-form-item-required')).toHaveLength(0); wrapper.find('input[type="checkbox"]').simulate('change', { target: { checked: true } }); wrapper.update(); expect(wrapper.find('.ant-form-item-required')).toHaveLength(1); }); it('should show related className when customize help', () => { const wrapper = mount(
, ); expect(wrapper.find('.ant-form-item-with-help').length).toBeTruthy(); }); it('warning when use v3 function', () => { Form.create(); expect(errorSpy).toHaveBeenCalledWith( 'Warning: [antd: Form] antd v4 removed `Form.create`. Please remove or use `@ant-design/compatible` instead.', ); }); // https://github.com/ant-design/ant-design/issues/20706 it('Error change should work', async () => { const wrapper = mount(
{ if (value === 'p') { return Promise.reject(new Error('not a p')); } return Promise.resolve(); }, }, ]} >
, ); /* eslint-disable no-await-in-loop */ for (let i = 0; i < 3; i += 1) { await change(wrapper, 0, ''); expect(wrapper.find('.ant-form-item-explain').first().text()).toEqual("'name' is required"); await change(wrapper, 0, 'p'); await delay(100); wrapper.update(); expect(wrapper.find('.ant-form-item-explain').first().text()).toEqual('not a p'); } /* eslint-enable */ }); // https://github.com/ant-design/ant-design/issues/20813 it('should update help directly when provided', () => { function App() { const [message, updateMessage] = React.useState(''); return (
); }; const wrapper = mount(); wrapper.find('button').simulate('click'); expect(errorSpy).not.toHaveBeenCalled(); }); it('return same form instance', () => { const instances = new Set(); const App = () => { const [form] = Form.useForm(); instances.add(form); const [, forceUpdate] = React.useState({}); return ( ); }; const wrapper = mount(); for (let i = 0; i < 5; i += 1) { wrapper.find('button').simulate('click'); } expect(instances.size).toEqual(1); }); it('avoid re-render', async () => { let renderTimes = 0; const MyInput = ({ value = '', ...props }) => { renderTimes += 1; return ; }; const Demo = () => (
); const wrapper = mount(); renderTimes = 0; wrapper.find('input').simulate('change', { target: { value: 'a', }, }); await delay(); expect(renderTimes).toEqual(1); expect(wrapper.find('input').props().value).toEqual('a'); }); it('warning with `defaultValue`', () => { mount(
, ); expect(errorSpy).toHaveBeenCalledWith( 'Warning: [antd: Form.Item] `defaultValue` will not work on controlled Field. You should use `initialValues` of Form instead.', ); }); });