From 1893c091b0b33849d264bd77e94c5f8a6d8713ba Mon Sep 17 00:00:00 2001 From: Corbin Robb Date: Wed, 9 Feb 2022 11:31:51 -0700 Subject: [PATCH 1/3] add sorting for query results grid and add natural order sort --- .../FilterableTable/FilterableTable.tsx | 63 ++++++++++++++----- .../FilterableTableStyles.less | 1 + 2 files changed, 50 insertions(+), 14 deletions(-) diff --git a/superset-frontend/src/components/FilterableTable/FilterableTable.tsx b/superset-frontend/src/components/FilterableTable/FilterableTable.tsx index 29a40d320d387..89f8080cc8ca1 100644 --- a/superset-frontend/src/components/FilterableTable/FilterableTable.tsx +++ b/superset-frontend/src/components/FilterableTable/FilterableTable.tsx @@ -114,8 +114,9 @@ interface FilterableTableProps { interface FilterableTableState { sortBy?: string; - sortDirection: SortDirectionType; + sortDirection?: SortDirectionType; fitted: boolean; + displayedList: Datum[]; } export default class FilterableTable extends PureComponent< @@ -175,8 +176,8 @@ export default class FilterableTable extends PureComponent< this.totalTableHeight = props.height; this.state = { - sortDirection: SortDirection.ASC, fitted: false, + displayedList: [...this.list], }; this.container = React.createRef(); @@ -191,7 +192,7 @@ export default class FilterableTable extends PureComponent< } getWidthsForColumns() { - const PADDING = 40; // accounts for cell padding and width of sorting icon + const PADDING = 50; // accounts for cell padding and width of sorting icon const widthsByColumnKey = {}; const cellContent = ([] as string[]).concat( ...this.props.orderedColumnKeys.map(key => { @@ -295,7 +296,30 @@ export default class FilterableTable extends PureComponent< sortBy: string; sortDirection: SortDirectionType; }) { - this.setState({ sortBy, sortDirection }); + this.setState( + ({ + sortBy: prevSortBy, + sortDirection: prevSortDirection, + displayedList: prevDisplayedList, + }) => { + const shouldClearSort = + prevSortDirection === SortDirection.DESC && prevSortBy === sortBy; + if (shouldClearSort) + return { + sortBy: undefined, + sortDirection: undefined, + displayedList: [...this.list], + }; + + return { + sortBy, + sortDirection, + displayedList: prevDisplayedList.sort( + this.sortResults(sortBy, sortDirection === SortDirection.DESC), + ), + }; + }, + ); } fitTableToWidthIfNeeded() { @@ -362,6 +386,17 @@ export default class FilterableTable extends PureComponent< }; } + sortGrid = (label: string) => { + this.sort({ + sortBy: label, + sortDirection: + this.state.sortDirection === SortDirection.DESC || + this.state.sortBy !== label + ? SortDirection.ASC + : SortDirection.DESC, + }); + }; + renderTableHeader({ dataKey, label, @@ -425,8 +460,14 @@ export default class FilterableTable extends PureComponent< : style.top, }} className={`${className} grid-cell grid-header-cell`} + role="columnheader" + tabIndex={columnIndex} + onClick={() => this.sortGrid(label)} > -
{label}
+ {label} + {this.state.sortBy === label && ( + + )} ); @@ -444,7 +485,7 @@ export default class FilterableTable extends PureComponent< style: React.CSSProperties; }) { const columnKey = this.props.orderedColumnKeys[columnIndex]; - const cellData = this.list[rowIndex][columnKey]; + const cellData = this.state.displayedList[rowIndex][columnKey]; const cellText = this.getCellContent({ cellData, columnKey }); const content = cellData === null ? {cellText} : cellText; @@ -563,19 +604,13 @@ export default class FilterableTable extends PureComponent< rowHeight, } = this.props; - let sortedAndFilteredList = this.list; + let sortedAndFilteredList = this.state.displayedList; // filter list if (filterText) { - sortedAndFilteredList = this.list.filter((row: Datum) => + sortedAndFilteredList = sortedAndFilteredList.filter((row: Datum) => this.hasMatch(filterText, row), ); } - // sort list - if (sortBy) { - sortedAndFilteredList = sortedAndFilteredList.sort( - this.sortResults(sortBy, sortDirection === SortDirection.DESC), - ); - } let { height } = this.props; let totalTableHeight = height; diff --git a/superset-frontend/src/components/FilterableTable/FilterableTableStyles.less b/superset-frontend/src/components/FilterableTable/FilterableTableStyles.less index 00786db70c9f3..d921149ae5fa5 100644 --- a/superset-frontend/src/components/FilterableTable/FilterableTableStyles.less +++ b/superset-frontend/src/components/FilterableTable/FilterableTableStyles.less @@ -57,6 +57,7 @@ .grid-header-cell { font-weight: @font-weight-bold; + cursor: pointer; } .ReactVirtualized__Table__headerColumn:last-of-type, From 63ba0541dfb2ca86fc375d1adf096a5fd033363d Mon Sep 17 00:00:00 2001 From: Corbin Robb Date: Wed, 9 Feb 2022 19:28:58 -0700 Subject: [PATCH 2/3] modify tests for natural/original order state --- .../FilterableTable/FilterableTable.test.tsx | 48 +++++++++---------- 1 file changed, 24 insertions(+), 24 deletions(-) diff --git a/superset-frontend/src/components/FilterableTable/FilterableTable.test.tsx b/superset-frontend/src/components/FilterableTable/FilterableTable.test.tsx index fad9e8245df22..3cb58ce6d06c9 100644 --- a/superset-frontend/src/components/FilterableTable/FilterableTable.test.tsx +++ b/superset-frontend/src/components/FilterableTable/FilterableTable.test.tsx @@ -116,10 +116,10 @@ describe('FilterableTable sorting - RTL', () => { expect(gridCells[1]).toHaveTextContent('Bravo'); expect(gridCells[2]).toHaveTextContent('Alpha'); - // Third click to sort ascending again + // Third click to clear sorting userEvent.click(stringColumn); - expect(gridCells[0]).toHaveTextContent('Alpha'); - expect(gridCells[1]).toHaveTextContent('Bravo'); + expect(gridCells[0]).toHaveTextContent('Bravo'); + expect(gridCells[1]).toHaveTextContent('Alpha'); expect(gridCells[2]).toHaveTextContent('Charlie'); }); @@ -151,10 +151,10 @@ describe('FilterableTable sorting - RTL', () => { expect(gridCells[1]).toHaveTextContent('10'); expect(gridCells[2]).toHaveTextContent('0'); - // Third click to sort ascending again + // Third click to clear sorting userEvent.click(integerColumn); - expect(gridCells[0]).toHaveTextContent('0'); - expect(gridCells[1]).toHaveTextContent('10'); + expect(gridCells[0]).toHaveTextContent('10'); + expect(gridCells[1]).toHaveTextContent('0'); expect(gridCells[2]).toHaveTextContent('100'); }); @@ -186,10 +186,10 @@ describe('FilterableTable sorting - RTL', () => { expect(gridCells[1]).toHaveTextContent('45.67'); expect(gridCells[2]).toHaveTextContent('1.23'); - // Third click to sort ascending again + // Third click to clear sorting userEvent.click(floatColumn); - expect(gridCells[0]).toHaveTextContent('1.23'); - expect(gridCells[1]).toHaveTextContent('45.67'); + expect(gridCells[0]).toHaveTextContent('45.67'); + expect(gridCells[1]).toHaveTextContent('1.23'); expect(gridCells[2]).toHaveTextContent('89.0000001'); }); @@ -257,17 +257,17 @@ describe('FilterableTable sorting - RTL', () => { expect(gridCells[9]).toHaveTextContent('26260.210000000003'); expect(gridCells[10]).toHaveTextContent('24078.610000000004'); - // Third click to sort ascending again + // Third click to clear sorting userEvent.click(mixedFloatColumn); - expect(gridCells[0]).toHaveTextContent('24078.610000000004'); - expect(gridCells[1]).toHaveTextContent('26260.210000000003'); - expect(gridCells[2]).toHaveTextContent('28550.59'); - expect(gridCells[3]).toHaveTextContent('48710.92'); - expect(gridCells[4]).toHaveTextContent('72212.86'); - expect(gridCells[5]).toHaveTextContent('98089.08000000002'); - expect(gridCells[6]).toHaveTextContent('144729.96000000002'); - expect(gridCells[7]).toHaveTextContent('145776.56'); - expect(gridCells[8]).toHaveTextContent('152718.97999999998'); + expect(gridCells[0]).toHaveTextContent('48710.92'); + expect(gridCells[1]).toHaveTextContent('145776.56'); + expect(gridCells[2]).toHaveTextContent('72212.86'); + expect(gridCells[3]).toHaveTextContent('144729.96000000002'); + expect(gridCells[4]).toHaveTextContent('26260.210000000003'); + expect(gridCells[5]).toHaveTextContent('152718.97999999998'); + expect(gridCells[6]).toHaveTextContent('28550.59'); + expect(gridCells[7]).toHaveTextContent('24078.610000000004'); + expect(gridCells[8]).toHaveTextContent('98089.08000000002'); expect(gridCells[9]).toHaveTextContent('3439718.0300000007'); expect(gridCells[10]).toHaveTextContent('4528047.219999993'); }); @@ -320,14 +320,14 @@ describe('FilterableTable sorting - RTL', () => { expect(gridCells[5]).toHaveTextContent('2021-01-02'); expect(gridCells[6]).toHaveTextContent('2021-01-01'); - // Third click to sort ascending again + // Third click to clear sorting userEvent.click(dsColumn); expect(gridCells[0]).toHaveTextContent('2021-01-01'); - expect(gridCells[1]).toHaveTextContent('2021-01-02'); - expect(gridCells[2]).toHaveTextContent('2021-01-03'); - expect(gridCells[3]).toHaveTextContent('2021-10-01'); + expect(gridCells[1]).toHaveTextContent('2022-01-01'); + expect(gridCells[2]).toHaveTextContent('2021-01-02'); + expect(gridCells[3]).toHaveTextContent('2021-01-03'); expect(gridCells[4]).toHaveTextContent('2021-12-01'); - expect(gridCells[5]).toHaveTextContent('2022-01-01'); + expect(gridCells[5]).toHaveTextContent('2021-10-01'); expect(gridCells[6]).toHaveTextContent('2022-01-02'); }); }); From dda6bbfb9583555098996ae5b09c934b632d9a7b Mon Sep 17 00:00:00 2001 From: Corbin Robb Date: Fri, 11 Feb 2022 13:08:57 -0700 Subject: [PATCH 3/3] clean up setState --- .../FilterableTable/FilterableTable.tsx | 49 ++++++++++--------- 1 file changed, 25 insertions(+), 24 deletions(-) diff --git a/superset-frontend/src/components/FilterableTable/FilterableTable.tsx b/superset-frontend/src/components/FilterableTable/FilterableTable.tsx index 89f8080cc8ca1..23d68f96850c9 100644 --- a/superset-frontend/src/components/FilterableTable/FilterableTable.tsx +++ b/superset-frontend/src/components/FilterableTable/FilterableTable.tsx @@ -296,30 +296,31 @@ export default class FilterableTable extends PureComponent< sortBy: string; sortDirection: SortDirectionType; }) { - this.setState( - ({ - sortBy: prevSortBy, - sortDirection: prevSortDirection, - displayedList: prevDisplayedList, - }) => { - const shouldClearSort = - prevSortDirection === SortDirection.DESC && prevSortBy === sortBy; - if (shouldClearSort) - return { - sortBy: undefined, - sortDirection: undefined, - displayedList: [...this.list], - }; - - return { - sortBy, - sortDirection, - displayedList: prevDisplayedList.sort( - this.sortResults(sortBy, sortDirection === SortDirection.DESC), - ), - }; - }, - ); + let updatedState: FilterableTableState; + + const shouldClearSort = + this.state.sortDirection === SortDirection.DESC && + this.state.sortBy === sortBy; + + if (shouldClearSort) { + updatedState = { + ...this.state, + sortBy: undefined, + sortDirection: undefined, + displayedList: [...this.list], + }; + } else { + updatedState = { + ...this.state, + sortBy, + sortDirection, + displayedList: [...this.list].sort( + this.sortResults(sortBy, sortDirection === SortDirection.DESC), + ), + }; + } + + this.setState(updatedState); } fitTableToWidthIfNeeded() {