/* eslint-disable react/no-string-refs, react/prefer-es6-class */ import produce from 'immer'; import { cloneDeep } from 'lodash'; import React from 'react'; import { act } from 'react-dom/test-utils'; import Upload from '..'; import mountTest from '../../../tests/shared/mountTest'; import rtlTest from '../../../tests/shared/rtlTest'; import { fireEvent, render, sleep } from '../../../tests/utils'; import Form from '../../form'; import { resetWarned } from '../../_util/warning'; import { getFileItem, isImageUrl, removeFileItem } from '../utils'; import { setup, teardown } from './mock'; globalThis.IS_REACT_ACT_ENVIRONMENT = true; describe('Upload', () => { mountTest(Upload); rtlTest(Upload); beforeEach(() => setup()); afterEach(() => teardown()); // Mock for rc-util raf window.requestAnimationFrame = callback => { window.setTimeout(callback, 16); }; window.cancelAnimationFrame = id => { window.clearTimeout(id); }; // https://github.com/react-component/upload/issues/36 it('should get refs inside Upload in componentDidMount', () => { let ref; class App extends React.Component { componentDidMount() { ref = this.refs.input; } render() { return ( ); } } render(); expect(ref).toBeDefined(); }); it('return promise in beforeUpload', async () => { jest.useFakeTimers(); const data = jest.fn(); const done = jest.fn(); const props = { action: 'http://upload.com', beforeUpload: () => new Promise(resolve => { setTimeout(() => resolve('success'), 100); }), data, onChange: ({ file }) => { if (file.status !== 'uploading') { expect(data).toHaveBeenCalled(); done(); } }, }; const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [{ file: 'foo.png' }], }, }); act(() => { jest.runAllTimers(); }); await act(async () => { for (let i = 0; i < 4; i += 1) { // eslint-disable-next-line no-await-in-loop await Promise.resolve(); } }); expect(done).toHaveBeenCalled(); jest.useRealTimers(); }); it('beforeUpload can be falsy', async () => { jest.useFakeTimers(); const done = jest.fn(); const props = { action: 'http://upload.com', beforeUpload: false, onChange: ({ file }) => { if (file.status !== 'uploading') { done(); } }, }; const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [{ file: 'foo.png' }], }, }); await act(async () => { for (let i = 0; i < 4; i += 1) { // eslint-disable-next-line no-await-in-loop await Promise.resolve(); } }); expect(done).toHaveBeenCalled(); jest.useRealTimers(); }); it('upload promise return file in beforeUpload', async () => { jest.useFakeTimers(); const done = jest.fn(); const data = jest.fn(); const props = { action: 'http://upload.com', beforeUpload: file => new Promise(resolve => { setTimeout(() => { const result = file; result.name = 'test.png'; resolve(result); }, 100); }), data, onChange: ({ file }) => { if (file.status !== 'uploading') { expect(data).toHaveBeenCalled(); expect(file.name).toEqual('test.png'); done(); } }, }; const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [{ file: 'foo.png' }], }, }); act(() => { jest.runAllTimers(); }); await act(async () => { for (let i = 0; i < 4; i += 1) { // eslint-disable-next-line no-await-in-loop await Promise.resolve(); } }); expect(done).toHaveBeenCalled(); jest.useRealTimers(); }); it('should not stop upload when return value of beforeUpload is false', done => { const fileList = [ { uid: 'bar', name: 'bar.png', }, ]; const mockFile = new File(['foo'], 'foo.png', { type: 'image/png', }); const data = jest.fn(); const props = { action: 'http://upload.com', fileList, beforeUpload: () => false, data, onChange: ({ file, fileList: updatedFileList }) => { expect(file instanceof File).toBe(true); expect(updatedFileList.map(f => f.name)).toEqual(['bar.png', 'foo.png']); expect(data).not.toHaveBeenCalled(); done(); }, }; const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [mockFile], }, }); }); it('should not stop upload when return value of beforeUpload is not false', done => { const data = jest.fn(); const props = { action: 'http://upload.com', beforeUpload() {}, data, onChange: () => { expect(data).toHaveBeenCalled(); done(); }, }; const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [{ file: 'foo.png' }], }, }); }); // https://github.com/ant-design/ant-design/issues/14779 it('should contain input file control if upload button is hidden', () => { const { container, rerender } = render( , ); expect(container.querySelectorAll('input[type="file"]')).toHaveLength(1); rerender(); expect(container.querySelectorAll('input[type="file"]')).toHaveLength(1); }); // https://github.com/ant-design/ant-design/issues/14298 it('should not have id if upload children is null, avoid being triggered by label', () => { const Demo = ({ children }) => (
{children}
); const { container, rerender } = render(
upload
, ); expect(container.querySelector('input#upload')).toBeTruthy(); rerender(); expect(container.querySelector('input#upload')).toBeFalsy(); }); // https://github.com/ant-design/ant-design/issues/16478 it('should not have id if Upload is disabled, avoid being triggered by label', () => { const Demo = ({ disabled }) => (
upload
); const { container: wrapper, rerender } = render(); expect(wrapper.querySelectorAll('input#upload').length).toBe(1); rerender(); expect(wrapper.querySelectorAll('input#upload').length).toBe(0); }); // https://github.com/ant-design/ant-design/issues/24197 it('should not have id if upload.Dragger is disabled, avoid being triggered by label', () => { const Demo = ({ disabled }) => (
upload
); const { container: wrapper, rerender } = render(); expect(wrapper.querySelectorAll('input#upload').length).toBe(1); rerender(); expect(wrapper.querySelectorAll('input#upload').length).toBe(0); }); it('should be controlled by fileList', () => { jest.useFakeTimers(); const fileList = [ { uid: '-1', name: 'foo.png', status: 'done', url: 'http://www.baidu.com/xxx.png', }, ]; const ref = React.createRef(); const { rerender } = render(); expect(ref.current.fileList).toEqual([]); rerender(); jest.runAllTimers(); expect(ref.current.fileList).toEqual(fileList); jest.useRealTimers(); }); it('should be able to get uid at first', () => { const fileList = [ { name: 'foo.png', status: 'done', url: 'http://www.baidu.com/xxx.png', }, ]; render(); fileList.forEach(file => { expect(file.uid).toBeDefined(); }); }); describe('util', () => { it('should be able to get fileItem', () => { const file = { uid: '-1', name: 'item.jpg' }; const fileList = [ { uid: '-1', name: 'item.jpg', }, ]; const targetItem = getFileItem(file, fileList); expect(targetItem).toBe(fileList[0]); }); it('should be able to remove fileItem', () => { const file = { uid: '-1', name: 'item.jpg' }; const fileList = [ { uid: '-1', name: 'item.jpg', }, { uid: '-2', name: 'item2.jpg', }, ]; const targetItem = removeFileItem(file, fileList); expect(targetItem).toEqual(fileList.slice(1)); }); it('remove fileItem and fileList with immutable data', () => { const file = { uid: '-3', name: 'item3.jpg' }; const fileList = produce( [ { uid: '-1', name: 'item.jpg', }, { uid: '-2', name: 'item2.jpg', }, ], draftState => { draftState.push({ uid: '-3', name: 'item3.jpg', }); }, ); const targetItem = removeFileItem(file, fileList); expect(targetItem).toEqual(fileList.slice(0, 2)); }); it('should not be able to remove fileItem', () => { const file = { uid: '-3', name: 'item.jpg' }; const fileList = [ { uid: '-1', name: 'item.jpg', }, { uid: '-2', name: 'item2.jpg', }, ]; const targetItem = removeFileItem(file, fileList); expect(targetItem).toBe(null); }); it('isImageUrl should work correctly when file.url is null', () => { const file = { url: null, }; expect(isImageUrl(file)).toBe(true); }); }); it('should support linkProps as object', () => { const fileList = [ { uid: '-1', name: 'foo.png', status: 'done', url: 'http://www.baidu.com/xxx.png', linkProps: { download: 'image', rel: 'noopener', }, }, ]; const { container: wrapper } = render(); const linkNode = wrapper.querySelector('a.ant-upload-list-item-name'); expect(linkNode.getAttribute('download')).toBe('image'); expect(linkNode.getAttribute('rel')).toBe('noopener'); }); it('should support linkProps as json stringify', () => { const linkPropsString = JSON.stringify({ download: 'image', rel: 'noopener', }); const fileList = [ { uid: '-1', name: 'foo.png', status: 'done', url: 'http://www.baidu.com/xxx.png', linkProps: linkPropsString, }, ]; const { container: wrapper } = render(); const linkNode = wrapper.querySelector('a.ant-upload-list-item-name'); expect(linkNode.getAttribute('download')).toBe('image'); expect(linkNode.getAttribute('rel')).toBe('noopener'); }); it('should not stop remove when return value of onRemove is false', done => { const mockRemove = jest.fn(() => false); const props = { onRemove: mockRemove, fileList: [ { uid: '-1', name: 'foo.png', status: 'done', url: 'http://www.baidu.com/xxx.png', }, ], }; const { container: wrapper } = render(); fireEvent.click(wrapper.querySelector('div.ant-upload-list-item .anticon-delete')); setTimeout(() => { expect(mockRemove).toHaveBeenCalled(); expect(props.fileList).toHaveLength(1); expect(props.fileList[0].status).toBe('done'); done(); }); }); // https://github.com/ant-design/ant-design/issues/18902 it('should not abort uploading until return value of onRemove is resolved as true', async () => { const file = { uid: '-1', name: 'foo.png', status: 'uploading', url: 'http://www.baidu.com/xxx.png', }; let removePromise; const onRemove = () => new Promise(resolve => { expect(file.status).toBe('uploading'); removePromise = resolve; }); const onChange = jest.fn(); const { container } = render( , ); fireEvent.click(container.querySelector('div.ant-upload-list-item .anticon-delete')); // uploadStart is a batch work which we need wait for react act await act(async () => { await Promise.resolve(); }); // Delay return true for remove await sleep(100); await act(async () => { await removePromise(true); }); expect(onChange).toHaveBeenCalled(); expect(file.status).toBe('removed'); }); it('should not stop download when return use onDownload', done => { const mockRemove = jest.fn(() => false); const props = { onRemove: mockRemove, showUploadList: { showDownloadIcon: true, }, fileList: [ { uid: '-1', name: 'foo.png', status: 'done', url: 'http://www.baidu.com/xxx.png', }, ], }; const { container: wrapper } = render( {}} />); fireEvent.click(wrapper.querySelector('div.ant-upload-list-item .anticon-download')); setTimeout(() => { expect(props.fileList).toHaveLength(1); expect(props.fileList[0].status).toBe('done'); done(); }); }); // https://github.com/ant-design/ant-design/issues/14439 it('should allow call abort function through upload instance', () => { const ref = React.createRef(); render( , ); expect(typeof ref.current.upload.abort).toBe('function'); }); it('correct dragCls when type is drag', () => { const fileList = [{ status: 'uploading', uid: 'file' }]; const { container: wrapper } = render( , ); expect(wrapper.querySelectorAll('.ant-upload-drag-uploading').length).toBe(1); }); it('return when targetItem is null', () => { const fileList = [{ uid: 'file' }]; const ref = React.createRef(); render( , ); expect(ref.current.onSuccess('', { uid: 'fileItem' })).toBe(undefined); expect(ref.current.onProgress('', { uid: 'fileItem' })).toBe(undefined); expect(ref.current.onError('', '', { uid: 'fileItem' })).toBe(undefined); }); it('should replace file when targetItem already exists', () => { const fileList = [{ uid: 'file', name: 'file' }]; const ref = React.createRef(); const { unmount } = render( , ); const newFile = { uid: 'file', name: 'file1', }; act(() => { ref.current.onBatchStart([ { file: newFile, parsedFile: newFile, }, ]); }); expect(ref.current.fileList.length).toBe(1); expect(ref.current.fileList[0].originFileObj).toEqual({ name: 'file1', uid: 'file', }); unmount(); }); it('warning if set `value`', () => { resetWarned(); const errorSpy = jest.spyOn(console, 'error').mockImplementation(() => {}); render(); expect(errorSpy).toHaveBeenCalledWith( 'Warning: [antd: Upload] `value` is not a valid prop, do you mean `fileList`?', ); errorSpy.mockRestore(); }); it('should be treated as file but not an image', () => { const file = { status: 'done', uid: '-1', type: 'video/mp4', url: 'https://zos.alipayobjects.com/rmsportal/IQKRngzUuFzJzGzRJXUs.png', }; const { container: wrapper } = render(); expect(wrapper.querySelectorAll('img').length).toBe(0); }); // https://github.com/ant-design/ant-design/issues/25077 it('should support events', () => { const onClick = jest.fn(); const onMouseEnter = jest.fn(); const onMouseLeave = jest.fn(); const { container: wrapper } = render( , ); fireEvent.click(wrapper.querySelectorAll('.ant-upload')[1]); expect(onClick).toHaveBeenCalled(); fireEvent.mouseEnter(wrapper.querySelectorAll('.ant-upload')[1]); expect(onMouseEnter).toHaveBeenCalled(); fireEvent.mouseLeave(wrapper.querySelectorAll('.ant-upload')[1]); expect(onMouseLeave).toHaveBeenCalled(); }); // https://github.com/ant-design/ant-design/issues/26427 it('should sync file list with control mode', async () => { jest.useFakeTimers(); const done = jest.fn(); let callTimes = 0; const customRequest = jest.fn(async options => { // stop here to make sure new fileList has been set and passed to Upload // eslint-disable-next-line no-promise-executor-return await new Promise(resolve => setTimeout(resolve, 0)); options.onProgress({ percent: 0 }); const url = Promise.resolve('https://ant.design'); options.onProgress({ percent: 100 }); options.onSuccess({}, { ...options.file, url }); }); const Demo = () => { const [fileList, setFileList] = React.useState([]); const onChange = async e => { const newFileList = Array.isArray(e) ? e : e.fileList; setFileList(newFileList); const file = newFileList[0]; callTimes += 1; switch (callTimes) { case 1: case 2: expect(file).toEqual(expect.objectContaining({ status: 'uploading', percent: 0 })); break; case 3: expect(file).toEqual(expect.objectContaining({ status: 'uploading', percent: 100 })); break; case 4: expect(file).toEqual(expect.objectContaining({ status: 'done', percent: 100 })); break; default: // Do nothing } if (callTimes >= 4) { done(); } }; return ( ); }; const { container: wrapper } = render(); fireEvent.change(wrapper.querySelector('input'), { target: { files: [{ file: 'foo.png' }], }, }); await act(async () => { for (let i = 0; i < 3; i += 1) { // eslint-disable-next-line no-await-in-loop await Promise.resolve(); } }); await act(() => { jest.runAllTimers(); }); await act(async () => { await Promise.resolve(); }); expect(done).toHaveBeenCalled(); jest.useRealTimers(); }); describe('maxCount', () => { it('replace when only 1', async () => { const onChange = jest.fn(); const fileList = [ { uid: 'bar', name: 'bar.png', }, ]; const props = { action: 'http://upload.com', fileList, onChange, maxCount: 1, }; const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [ new File(['foo'], 'foo.png', { type: 'image/png', }), ], }, }); await sleep(20); expect(onChange.mock.calls[0][0].fileList).toHaveLength(1); expect(onChange.mock.calls[0][0].fileList[0]).toEqual( expect.objectContaining({ name: 'foo.png', }), ); }); it('maxCount > 1', async () => { const onChange = jest.fn(); const fileList = [ { uid: 'bar', name: 'bar.png', }, ]; const props = { action: 'http://upload.com', fileList, onChange, maxCount: 2, }; const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [ new File(['foo'], 'foo.png', { type: 'image/png', }), new File(['invisible'], 'invisible.png', { type: 'image/png', }), ], }, }); await sleep(20); expect(onChange.mock.calls[0][0].fileList).toHaveLength(2); expect(onChange.mock.calls[0][0].fileList).toEqual([ expect.objectContaining({ name: 'bar.png', }), expect.objectContaining({ name: 'foo.png', }), ]); }); }); it('auto fill file uid', () => { const fileList = [ { name: 'bamboo.png', }, ]; expect(fileList[0].uid).toBeFalsy(); render( , ); expect(fileList[0].uid).toBeTruthy(); }); it('Proxy should support deepClone', async () => { const onChange = jest.fn(); const { container: wrapper } = render( , ); fireEvent.change(wrapper.querySelector('input'), { target: { files: [ new File(['foo'], 'foo.png', { type: 'image/png', }), ], }, }); await sleep(); const { file } = onChange.mock.calls[0][0]; const clone = cloneDeep(file); expect(Object.getOwnPropertyDescriptor(file, 'name')).toEqual( expect.objectContaining({ value: 'foo.png' }), ); ['uid', 'name', 'lastModified', 'lastModifiedDate', 'size', 'type'].forEach(key => { expect(key in clone).toBeTruthy(); }); }); it('not break on freeze object', async () => { const fileList = [ { fileName: 'Test.png', name: 'SupportIS App - potwierdzenie.png', thumbUrl: null, downloadUrl: 'https://localhost:5001/api/files/ff2917ce-e4b9-4542-84da-31cdbe7c273f', status: 'done', }, ]; const frozenFileList = fileList.map(file => Object.freeze(file)); const { container: wrapper } = render(); const rmBtn = wrapper.querySelectorAll('.ant-upload-list-item-card-actions-btn'); fireEvent.click(rmBtn[rmBtn.length - 1]); // Wait for Upload async remove await act(async () => { await sleep(); }); }); // https://github.com/ant-design/ant-design/issues/30390 // IE11 Does not support the File constructor it('should not break in IE if beforeUpload returns false', async () => { const onChange = jest.fn(); const { container } = render( false} fileList={[]} onChange={onChange} />, ); const fileConstructor = () => { throw new TypeError("Object doesn't support this action"); }; jest.spyOn(global, 'File').mockImplementationOnce(fileConstructor); fireEvent.change(container.querySelector('input'), { target: { files: [{ file: 'foo.png' }], }, }); // React 18 is async now await sleep(); expect(onChange.mock.calls[0][0].fileList).toHaveLength(1); }); // https://github.com/ant-design/ant-design/issues/33819 it('should show the animation of the upload children leaving when the upload children becomes null', async () => { jest.useFakeTimers(); const { container, rerender } = render( , ); rerender(); expect(container.querySelector('.ant-upload-select-picture-card')).toHaveClass( 'ant-upload-animate-inline-leave-start', ); expect(container.querySelector('.ant-upload-select-picture-card')).toHaveStyle({ pointerEvents: 'none', }); // Motion leave status change: start > active await act(() => { jest.runAllTimers(); }); fireEvent.animationEnd(container.querySelector('.ant-upload-select-picture-card')); expect(container.querySelector('.ant-upload-select-picture-card')).not.toHaveClass( 'ant-upload-animate-inline-leave-start', ); jest.useRealTimers(); }); it(' should pass prefixCls', async () => { const { container: wrapper } = render(); expect(wrapper.querySelectorAll('.ant-upload-list').length).toBeGreaterThan(0); const { container: wrapper2 } = render(); expect(wrapper2.querySelectorAll('.custom-upload-list').length).toBeGreaterThan(0); }); });