/* eslint-disable react/no-multi-comp */ import React from 'react'; import Table from '..'; import { act, fireEvent, render } from '../../../tests/utils'; describe('Table.sorter', () => { const sorterFn = (a, b) => a.name[0].charCodeAt() - b.name[0].charCodeAt(); const column = { title: 'Name', dataIndex: 'name', key: 'name', sorter: sorterFn, }; const data = [ { key: 0, name: 'Jack' }, { key: 1, name: 'Lucy' }, { key: 2, name: 'Tom' }, { key: 3, name: 'Jerry' }, ]; function createTable(tableProps = {}, columnProps = {}) { return ( ); } function renderedNames(contain) { const namesList = []; contain .querySelector('.ant-table-tbody') .querySelectorAll('tr') .forEach(tr => { namesList.push(tr.querySelector('td').textContent); }); return namesList; } it('renders sorter icon correctly', () => { const { container } = render(createTable()); expect(container.querySelector('thead')).toMatchSnapshot(); }); it('default sort order ascend', () => { const { container } = render( createTable( {}, { defaultSortOrder: 'ascend', }, ), ); expect(renderedNames(container)).toEqual(['Jack', 'Jerry', 'Lucy', 'Tom']); }); it('default sort order descend', () => { const { container } = render( createTable( {}, { defaultSortOrder: 'descend', }, ), ); expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); expect(container.querySelector('th').getAttribute('aria-sort')).toEqual('descending'); }); it('should change aria-sort when default sort order is set to descend', () => { const { container } = render( createTable( { sortDirections: ['descend', 'ascend'], }, { defaultSortOrder: 'descend', }, ), ); const getNameColumn = () => container.querySelector('th'); // Test that it cycles through the order of sortDirections expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); fireEvent.click(container.querySelector('.ant-table-column-sorters')); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); fireEvent.click(container.querySelector('.ant-table-column-sorters')); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); }); it('sort records', () => { const { container } = render(createTable()); const getNameColumn = () => container.querySelector('th'); // first assert default state expect(renderedNames(container)).toEqual(['Jack', 'Lucy', 'Tom', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); // ascend fireEvent.click(container.querySelector('.ant-table-column-sorters')); expect(renderedNames(container)).toEqual(['Jack', 'Jerry', 'Lucy', 'Tom']); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); // descend fireEvent.click(container.querySelector('.ant-table-column-sorters')); expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); }); it('sort records with keydown', () => { const { container } = render(createTable()); // ascend fireEvent.keyDown(container.querySelector('.ant-table-column-sorters'), { keyCode: 13 }); expect(renderedNames(container)).toEqual(['Jack', 'Jerry', 'Lucy', 'Tom']); // descend fireEvent.keyDown(container.querySelector('.ant-table-column-sorters'), { keyCode: 13 }); expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); }); describe('can be controlled by sortOrder', () => { it('single', () => { const { container } = render( createTable({ columns: [{ ...column, sortOrder: 'ascend' }], }), ); expect(renderedNames(container)).toEqual(['Jack', 'Jerry', 'Lucy', 'Tom']); }); it('invalidate mix with single & multiple sorters', () => { const { container } = render( createTable({ columns: [ { title: 'Name', dataIndex: 'name', sortOrder: 'ascend', sorter: { multiple: 1, }, }, { title: 'Name', dataIndex: 'name', sortOrder: 'ascend', sorter: {}, }, ], }), ); expect(renderedNames(container)).toEqual(['Jack', 'Lucy', 'Tom', 'Jerry']); }); }); it('provides sortOrder in the sorterFn', () => { let actualSortOrder; render( createTable( {}, { sortOrder: 'ascend', sorter: (a, b, sortOrder) => { actualSortOrder = sortOrder; return sorterFn(a, b); }, }, ), ); expect(actualSortOrder).toEqual('ascend'); }); it('can update column sortOrder', () => { const { container, rerender } = render( createTable({ columns: [column], }), ); expect(renderedNames(container)).toEqual(['Jack', 'Lucy', 'Tom', 'Jerry']); rerender( createTable({ columns: [{ ...column, sortOrder: 'ascend' }], }), ); expect(renderedNames(container)).toEqual(['Jack', 'Jerry', 'Lucy', 'Tom']); }); it('fires change event', () => { const handleChange = jest.fn(); const { container } = render(createTable({ onChange: handleChange })); // ascent fireEvent.click(container.querySelector('.ant-table-column-sorters')); const sorter1 = handleChange.mock.calls[0][2]; expect(sorter1.column.dataIndex).toBe('name'); expect(sorter1.order).toBe('ascend'); expect(sorter1.field).toBe('name'); expect(sorter1.columnKey).toBe('name'); fireEvent.click(container.querySelector('.ant-table-column-sorters')); const sorter2 = handleChange.mock.calls[1][2]; expect(sorter2.column.dataIndex).toBe('name'); expect(sorter2.order).toBe('descend'); expect(sorter2.field).toBe('name'); expect(sorter2.columnKey).toBe('name'); fireEvent.click(container.querySelector('.ant-table-column-sorters')); const sorter3 = handleChange.mock.calls[2][2]; expect(sorter3.column).toBe(undefined); expect(sorter3.order).toBe(undefined); expect(sorter3.field).toBe('name'); expect(sorter3.columnKey).toBe('name'); }); it('hover header show sorter tooltip', () => { // tooltip has delay jest.useFakeTimers(); const { container, rerender } = render(createTable()); // default show sorter tooltip fireEvent.mouseEnter(container.querySelector('.ant-table-column-sorters')); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeTruthy(); fireEvent.mouseOut(container.querySelector('.ant-table-column-sorters')); // set table props showSorterTooltip is false rerender(createTable({ showSorterTooltip: false })); fireEvent.mouseEnter(container.querySelector('.ant-table-column-sorters')); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeFalsy(); fireEvent.mouseOut(container.querySelector('.ant-table-column-sorters')); // set table props showSorterTooltip is false, column showSorterTooltip is true rerender( createTable({ showSorterTooltip: true, columns: [{ ...column, showSorterTooltip: true }] }), ); fireEvent.mouseEnter(container.querySelector('.ant-table-column-sorters')); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeTruthy(); fireEvent.mouseOut(container.querySelector('.ant-table-column-sorters')); // set table props showSorterTooltip is true, column showSorterTooltip is false rerender( createTable({ showSorterTooltip: true, columns: [{ ...column, showSorterTooltip: false }], }), ); fireEvent.mouseEnter(container.querySelector('.ant-table-column-sorters')); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeFalsy(); fireEvent.mouseOut(container.querySelector('.ant-table-column-sorters')); }); it('should show correct tooltip when showSorterTooltip is an object', () => { // basically copied from 'hover header show sorter tooltip' jest.useFakeTimers(); const { container, rerender } = render( createTable({ showSorterTooltip: { placement: 'bottom', title: 'static title' } }), ); fireEvent.mouseEnter(container.querySelector('.ant-table-column-sorters')); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeTruthy(); fireEvent.mouseOut(container.querySelector('.ant-table-column-sorters')); // Root to false rerender(createTable({ showSorterTooltip: false })); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeFalsy(); // Column to true rerender( createTable({ showSorterTooltip: false, columns: [{ ...column, showSorterTooltip: true }], }), ); fireEvent.mouseEnter(container.querySelector('.ant-table-column-sorters')); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeTruthy(); fireEvent.mouseOut(container.querySelector('.ant-table-column-sorters')); // Column to false rerender( createTable({ showSorterTooltip: true, columns: [{ ...column, showSorterTooltip: false }], }), ); act(() => { jest.runAllTimers(); }); expect(container.querySelector('.ant-tooltip-open')).toBeFalsy(); }); it('works with grouping columns in controlled mode', () => { const columns = [ { title: 'group', key: 'group', children: [ { title: 'Name', dataIndex: 'name', key: 'name', sorter: sorterFn, sortOrder: 'descend', }, { title: 'Age', dataIndex: 'age', key: 'age', }, ], }, ]; const testData = [ { key: 0, name: 'Jack', age: 11 }, { key: 1, name: 'Lucy', age: 20 }, { key: 2, name: 'Tom', age: 21 }, { key: 3, name: 'Jerry', age: 22 }, ]; const { container } = render(
); expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); }); // https://github.com/ant-design/ant-design/issues/11246#issuecomment-405009167 it('Allow column title as render props with sortOrder argument', () => { const title = ({ sortOrder }) =>
{sortOrder}
; const columns = [ { title, key: 'group', sorter: true, }, ]; const testData = [ { key: 0, name: 'Jack', age: 11 }, { key: 1, name: 'Lucy', age: 20 }, { key: 2, name: 'Tom', age: 21 }, { key: 3, name: 'Jerry', age: 22 }, ]; const { container } = render(
); expect(container.querySelector('.custom-title').textContent).toEqual(''); fireEvent.click(container.querySelector('.ant-table-column-sorters')); expect(container.querySelector('.custom-title').textContent).toEqual('ascend'); fireEvent.click(container.querySelector('.ant-table-column-sorters')); expect(container.querySelector('.custom-title').textContent).toEqual('descend'); }); // https://github.com/ant-design/ant-design/pull/12264#discussion_r218053034 it('should sort from beginning state when toggle from different columns', () => { const columns = [ { title: 'name', dataIndex: 'name', sorter: true, }, { title: 'age', dataIndex: 'age', sorter: true, }, ]; const testData = [ { key: 0, name: 'Jack', age: 11 }, { key: 1, name: 'Lucy', age: 20 }, { key: 2, name: 'Tom', age: 21 }, { key: 3, name: 'Jerry', age: 22 }, ]; const { container } = render(
); const getNameColumn = () => container.querySelectorAll('.ant-table-column-has-sorters')[0]; const getAgeColumn = () => container.querySelectorAll('.ant-table-column-has-sorters')[1]; const getNameIcon = name => getNameColumn().querySelector(`.ant-table-column-sorter-${name}`); const getAgeIcon = name => getAgeColumn().querySelector(`.ant-table-column-sorter-${name}`); // sort name fireEvent.click(getNameColumn()); expect(getNameIcon('up').className.includes('active')).toBeTruthy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); expect(getAgeIcon('up').className.includes('active')).toBeFalsy(); expect(getAgeColumn().getAttribute('aria-sort')).toEqual(null); // sort age fireEvent.click(getAgeColumn()); expect(getNameIcon('up').className.includes('active')).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); expect(getAgeIcon('up').className.includes('active')).toBeTruthy(); expect(getAgeColumn().getAttribute('aria-sort')).toEqual('ascending'); }); // https://github.com/ant-design/ant-design/issues/12571 it('should toggle sort state when columns are put in render', () => { const testData = [ { key: 0, name: 'Jack', age: 11 }, { key: 1, name: 'Lucy', age: 20 }, { key: 2, name: 'Tom', age: 21 }, { key: 3, name: 'Jerry', age: 22 }, ]; class TableTest extends React.Component { state = { pagination: {}, }; onChange = pagination => { this.setState({ pagination }); }; render() { const columns = [ { title: 'name', dataIndex: 'name', sorter: true, }, ]; const { pagination } = this.state; return (
); } } const { container } = render(); const getNameColumn = () => container.querySelector('th'); const getIcon = name => getNameColumn().querySelector(`.ant-table-column-sorter-${name}`); expect(getIcon('up').className.includes('active')).toBeFalsy(); expect(getIcon('down').className.includes('active')).toBeFalsy(); // sort name fireEvent.click(getNameColumn()); expect(getIcon('up').className.includes('active')).toBeTruthy(); expect(getIcon('down').className.includes('active')).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); // sort name fireEvent.click(getNameColumn()); expect(getIcon('up').className.includes('active')).toBeFalsy(); expect(getIcon('down').className.includes('active')).toBeTruthy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); // sort name fireEvent.click(getNameColumn()); expect(getIcon('up').className.includes('active')).toBeFalsy(); expect(getIcon('down').className.includes('active')).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); }); // https://github.com/ant-design/ant-design/issues/12737 // https://github.com/ant-design/ant-design/issues/19398 it('should toggle sort state when columns with non primitive properties are put in render', () => { const testData = [ { key: 0, name: 'Jack', age: 11 }, { key: 1, name: 'Lucy', age: 20 }, { key: 2, name: 'Tom', age: 21 }, { key: 3, name: 'Jerry', age: 22 }, ]; class TableTest extends React.Component { state = { pagination: {}, }; onChange = pagination => { this.setState({ pagination }); }; render() { const columns = [ { title: 'name', dataIndex: 'name', sorter: true, render: text => text, array: ['1', '2', 3], }, ]; const { pagination } = this.state; return (
); } } const { container } = render(); const getNameColumn = () => container.querySelector('th'); const getIcon = name => getNameColumn().querySelector(`.ant-table-column-sorter-${name}`); expect(getIcon('up').className.includes('active')).toBeFalsy(); expect(getIcon('down').className.includes('active')).toBeFalsy(); // sort name fireEvent.click(getNameColumn()); expect(getIcon('up').className.includes('active')).toBeTruthy(); expect(getIcon('down').className.includes('active')).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); // sort name fireEvent.click(getNameColumn()); expect(getIcon('up').className.includes('active')).toBeFalsy(); expect(getIcon('down').className.includes('active')).toBeTruthy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); // sort name fireEvent.click(getNameColumn()); expect(getIcon('up').className.includes('active')).toBeFalsy(); expect(getIcon('down').className.includes('active')).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); }); // https://github.com/ant-design/ant-design/issues/12870 it('should toggle sort state when columns with key are put in render', () => { const testData = [ { key: 0, name: 'Jack', age: 11 }, { key: 1, name: 'Lucy', age: 20 }, { key: 2, name: 'Tom', age: 21 }, { key: 3, name: 'Jerry', age: 22 }, ]; class TableTest extends React.Component { state = { pagination: {}, }; onChange = pagination => { this.setState({ pagination }); }; render() { const columns = [ { title: 'name', dataIndex: 'name', sorter: true, key: 'a', style: { fontSize: 18, }, }, ]; const { pagination } = this.state; return (
); } } const { container } = render(); const getNameColumn = () => container.querySelector('th'); expect( getNameColumn().querySelector('.ant-table-column-sorter-up').className.includes('active'), ).toBeFalsy(); expect( getNameColumn().querySelector('.ant-table-column-sorter-down').className.includes('active'), ).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); // sort name fireEvent.click(getNameColumn()); expect( getNameColumn().querySelector('.ant-table-column-sorter-up').className.includes('active'), ).toBeTruthy(); expect( getNameColumn().querySelector('.ant-table-column-sorter-down').className.includes('active'), ).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); // sort name fireEvent.click(getNameColumn()); expect( getNameColumn().querySelector('.ant-table-column-sorter-up').className.includes('active'), ).toBeFalsy(); expect( getNameColumn().querySelector('.ant-table-column-sorter-down').className.includes('active'), ).toBeTruthy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); // sort name fireEvent.click(getNameColumn()); expect( getNameColumn().querySelector('.ant-table-column-sorter-up').className.includes('active'), ).toBeFalsy(); expect( getNameColumn().querySelector('.ant-table-column-sorter-down').className.includes('active'), ).toBeFalsy(); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); }); it('should first sort by descend, then ascend, then cancel sort', () => { const { container } = render( createTable({ sortDirections: ['descend', 'ascend'], }), ); const getNameColumn = () => container.querySelector('th'); // descend fireEvent.click(getNameColumn()); expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); // ascend fireEvent.click(getNameColumn()); expect(renderedNames(container)).toEqual(['Jack', 'Jerry', 'Lucy', 'Tom']); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); // cancel sort fireEvent.click(getNameColumn()); expect(renderedNames(container)).toEqual(['Jack', 'Lucy', 'Tom', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); }); it('should first sort by descend, then cancel sort', () => { const { container } = render( createTable({ sortDirections: ['descend'], }), ); const getNameColumn = () => container.querySelector('th'); // default expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); // descend fireEvent.click(getNameColumn()); expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); // cancel sort fireEvent.click(getNameColumn()); expect(renderedNames(container)).toEqual(['Jack', 'Lucy', 'Tom', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); }); it('should first sort by descend, then cancel sort. (column prop)', () => { const { container } = render( createTable( {}, { sortDirections: ['descend'], }, ), ); const getNameColumn = () => container.querySelector('th'); // default expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); // descend fireEvent.click(getNameColumn()); expect(renderedNames(container)).toEqual(['Tom', 'Lucy', 'Jack', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual('descending'); // cancel sort fireEvent.click(getNameColumn()); expect(renderedNames(container)).toEqual(['Jack', 'Lucy', 'Tom', 'Jerry']); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); }); it('pagination back', () => { const onPageChange = jest.fn(); const onChange = jest.fn(); const { container } = render( createTable({ pagination: { pageSize: 2, defaultCurrent: 2, onChange: onPageChange, }, onChange, }), ); const getNameColumn = () => container.querySelector('th'); expect(getNameColumn().getAttribute('aria-sort')).toEqual(null); fireEvent.click(getNameColumn()); expect(onChange.mock.calls[0][0].current).toBe(2); expect(onPageChange).not.toHaveBeenCalled(); expect(getNameColumn().getAttribute('aria-sort')).toEqual('ascending'); }); it('should support onHeaderCell in sort column', () => { const onClick = jest.fn(); const { container } = render(
({ onClick }), sorter: true }]} />, ); fireEvent.click(container.querySelector('th')); expect(onClick).toHaveBeenCalled(); }); it('could sort data with children', () => { const { container } = render( createTable( { defaultExpandAllRows: true, dataSource: [ { key: '1', name: 'Brown', children: [ { key: '2', name: 'Zoe', }, { key: '3', name: 'Mike', children: [ { key: '3-1', name: 'Petter', }, { key: '3-2', name: 'Alex', }, ], }, { key: '4', name: 'Green', }, ], }, ], }, { defaultSortOrder: 'ascend', }, ), ); expect(renderedNames(container)).toEqual(['Brown', 'Green', 'Mike', 'Alex', 'Petter', 'Zoe']); }); // https://github.com/ant-design/ant-design/issues/19443 it('should not being inifinite loop when using Table.Column with sortOrder', () => { class Demo extends React.Component { componentDidMount() { this.setState({}); } render() { return (
); } } expect(() => { render(); }).not.toThrow(); }); it('should support defaultOrder in Column', () => { const { asFragment } = render(
, ); expect(asFragment().firstChild).toMatchSnapshot(); }); // https://github.com/ant-design/ant-design/issues/20096 it('invalidate sorter should not display sorter button', () => { const { container } = render( , ); expect(container.querySelectorAll('.ant-table-column-sorter-inner')).toHaveLength(0); }); // https://github.com/ant-design/ant-design/issues/21193 it('table with sugar column', () => { const { container } = render(
a.chinese - b.chinese, multiple: 3, }} /> a.math - b.math, multiple: 2, }} />
, ); fireEvent.click(container.querySelector('th')); expect(container.querySelectorAll('th.ant-table-column-sort')).toHaveLength(1); }); it('surger should support sorterOrder', () => { const { container } = render(
, ); expect( container.querySelector('.ant-table-column-sorter-up').className.includes('active'), ).toBeTruthy(); expect( container.querySelector('.ant-table-column-sorter-down').className.includes('active'), ).toBeFalsy(); }); it('controlled multiple group', () => { const groupColumns = [ { title: 'Math Score', dataIndex: 'math1', sortOrder: 'ascend', sorter: { multiple: 1 }, children: [ { title: 'math', dataIndex: 'math', }, ], }, { title: 'English Score', dataIndex: 'english', sortOrder: 'descend', sorter: { multiple: 2 }, }, ]; const groupData = [ { key: '1', name: 'John Brown', chinese: 98, math: 60, english: 70, }, ]; const { container } = render(); expect( container .querySelectorAll('.ant-table-column-sorter-full')[0] .querySelector('.ant-table-column-sorter-up') .className.includes('active'), ).toBeTruthy(); expect( container .querySelectorAll('.ant-table-column-sorter-full')[1] .querySelector('.ant-table-column-sorter-down') .className.includes('active'), ).toBeTruthy(); }); it('onChange with correct sorter for multiple', () => { const groupColumns = [ { title: 'Math Score', dataIndex: 'math', sorter: { multiple: 1 }, }, { title: 'English Score', dataIndex: 'english', sorter: { multiple: 2 }, }, ]; const groupData = [ { key: '1', name: 'John Brown', chinese: 98, math: 60, english: 70, }, ]; const onChange = jest.fn(); const { container } = render(
, ); function clickToMatchExpect(index, sorter) { fireEvent.click(container.querySelectorAll('.ant-table-column-sorters')[index]); expect(onChange).toHaveBeenCalledWith( expect.anything(), expect.anything(), expect.objectContaining(sorter), expect.anything(), ); onChange.mockReset(); } // First clickToMatchExpect(0, { field: 'math', order: 'ascend' }); clickToMatchExpect(0, { field: 'math', order: 'descend' }); clickToMatchExpect(0, { field: 'math', order: undefined }); // Last clickToMatchExpect(1, { field: 'english', order: 'ascend' }); clickToMatchExpect(1, { field: 'english', order: 'descend' }); clickToMatchExpect(1, { field: 'english', order: undefined }); }); });