Skip to content

Commit a6a59d5

Browse files
committed
feat(components/sort): add multi-sort support
Adds multi-column sorting capability to MatSort, allowing to sort a table on multiple of its columns at once by toggling matSortMultiple. This feature adds a new sortState variable inside MatSort that should be used as a source of truth when matSortMultiple is enabled. Fixes #24102
1 parent 70d416f commit a6a59d5

File tree

9 files changed

+389
-96
lines changed

9 files changed

+389
-96
lines changed

src/components-examples/material/table/table-sorting/table-sorting-example.css

+4
Original file line numberDiff line numberDiff line change
@@ -5,3 +5,7 @@ table {
55
th.mat-sort-header-sorted {
66
color: black;
77
}
8+
9+
.example-sorting-toggle-group {
10+
margin: 8px;
11+
}

src/components-examples/material/table/table-sorting/table-sorting-example.html

+28-12
Original file line numberDiff line numberDiff line change
@@ -1,36 +1,52 @@
1+
<div>
2+
<mat-button-toggle-group #multiSorting="matButtonToggleGroup" class="example-sorting-toggle-group">
3+
<mat-button-toggle [value]="false">Single column sorting</mat-button-toggle>
4+
<mat-button-toggle [value]="true">Multi column sorting</mat-button-toggle>
5+
</mat-button-toggle-group>
6+
</div>
7+
18
<table mat-table [dataSource]="dataSource" matSort (matSortChange)="announceSortChange($event)"
9+
[matSortMultiple]="multiSorting.value"
210
class="mat-elevation-z8">
311

412
<!-- Position Column -->
5-
<ng-container matColumnDef="position">
13+
<ng-container matColumnDef="firstName">
614
<th mat-header-cell *matHeaderCellDef mat-sort-header sortActionDescription="Sort by number">
7-
No.
15+
First name
816
</th>
9-
<td mat-cell *matCellDef="let element"> {{element.position}} </td>
17+
<td mat-cell *matCellDef="let element"> {{element.firstName}} </td>
1018
</ng-container>
1119

1220
<!-- Name Column -->
13-
<ng-container matColumnDef="name">
21+
<ng-container matColumnDef="lastName">
1422
<th mat-header-cell *matHeaderCellDef mat-sort-header sortActionDescription="Sort by name">
15-
Name
23+
Last name
1624
</th>
17-
<td mat-cell *matCellDef="let element"> {{element.name}} </td>
25+
<td mat-cell *matCellDef="let element"> {{element.lastName}} </td>
1826
</ng-container>
1927

2028
<!-- Weight Column -->
21-
<ng-container matColumnDef="weight">
29+
<ng-container matColumnDef="position">
2230
<th mat-header-cell *matHeaderCellDef mat-sort-header sortActionDescription="Sort by weight">
23-
Weight
31+
Position
32+
</th>
33+
<td mat-cell *matCellDef="let element"> {{element.position}} </td>
34+
</ng-container>
35+
36+
<!-- Symbol Column -->
37+
<ng-container matColumnDef="office">
38+
<th mat-header-cell *matHeaderCellDef mat-sort-header sortActionDescription="Sort by symbol">
39+
Office
2440
</th>
25-
<td mat-cell *matCellDef="let element"> {{element.weight}} </td>
41+
<td mat-cell *matCellDef="let element"> {{element.office}} </td>
2642
</ng-container>
2743

2844
<!-- Symbol Column -->
29-
<ng-container matColumnDef="symbol">
45+
<ng-container matColumnDef="salary">
3046
<th mat-header-cell *matHeaderCellDef mat-sort-header sortActionDescription="Sort by symbol">
31-
Symbol
47+
Salary
3248
</th>
33-
<td mat-cell *matCellDef="let element"> {{element.symbol}} </td>
49+
<td mat-cell *matCellDef="let element"> {{element.salary}} </td>
3450
</ng-container>
3551

3652
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>

src/components-examples/material/table/table-sorting/table-sorting-example.ts

+29-20
Original file line numberDiff line numberDiff line change
@@ -2,25 +2,34 @@ import {LiveAnnouncer} from '@angular/cdk/a11y';
22
import {AfterViewInit, Component, ViewChild} from '@angular/core';
33
import {MatSort, Sort, MatSortModule} from '@angular/material/sort';
44
import {MatTableDataSource, MatTableModule} from '@angular/material/table';
5+
import {MatButtonToggle, MatButtonToggleGroup} from '@angular/material/button-toggle';
6+
import {MatButton} from '@angular/material/button';
57

6-
export interface PeriodicElement {
7-
name: string;
8-
position: number;
9-
weight: number;
10-
symbol: string;
8+
export interface EmployeeData {
9+
firstName: string;
10+
lastName: string;
11+
position: string;
12+
office: string;
13+
salary: number;
1114
}
12-
const ELEMENT_DATA: PeriodicElement[] = [
13-
{position: 1, name: 'Hydrogen', weight: 1.0079, symbol: 'H'},
14-
{position: 2, name: 'Helium', weight: 4.0026, symbol: 'He'},
15-
{position: 3, name: 'Lithium', weight: 6.941, symbol: 'Li'},
16-
{position: 4, name: 'Beryllium', weight: 9.0122, symbol: 'Be'},
17-
{position: 5, name: 'Boron', weight: 10.811, symbol: 'B'},
18-
{position: 6, name: 'Carbon', weight: 12.0107, symbol: 'C'},
19-
{position: 7, name: 'Nitrogen', weight: 14.0067, symbol: 'N'},
20-
{position: 8, name: 'Oxygen', weight: 15.9994, symbol: 'O'},
21-
{position: 9, name: 'Fluorine', weight: 18.9984, symbol: 'F'},
22-
{position: 10, name: 'Neon', weight: 20.1797, symbol: 'Ne'},
23-
];
15+
16+
const MULTI_SORT_DATA: EmployeeData[] = [
17+
{firstName: "Garrett", lastName: "Winters", position: "Accountant", office: "Tokyo", salary: 170750},
18+
{firstName: "Airi", lastName: "Satou", position: "Accountant", office: "Tokyo", salary: 162700},
19+
{firstName: "Donna", lastName: "Snider", position: "Customer Support", office: "New York", salary: 112000},
20+
{firstName: "Serge", lastName: "Baldwin", position: "Data Coordinator", office: "Singapore", salary: 138575},
21+
{firstName: "Thor", lastName: "Walton", position: "Developer", office: "New York", salary: 98540},
22+
{firstName: "Gavin", lastName: "Joyce", position: "Developer", office: "Edinburgh", salary: 92575},
23+
{firstName: "Suki", lastName: "Burks", position: "Developer", office: "London", salary: 114500},
24+
{firstName: "Jonas", lastName: "Alexander", position: "Developer", office: "San Francisco", salary: 86500},
25+
{firstName: "Jackson", lastName: "Bradshaw", position: "Director", office: "New York", salary: 645750},
26+
{firstName: "Brielle", lastName: "Williamson", position: "Integration Specialist", office: "New York", salary: 372000},
27+
{firstName: "Michelle", lastName: "House", position: "Integration Specialist", office: "Sydney", salary: 95400},
28+
{firstName: "Michael", lastName: "Bruce", position: "Javascript Developer", office: "Singapore", salary: 183000},
29+
{firstName: "Ashton", lastName: "Cox", position: "Junior Technical Author", office: "San Francisco", salary: 86000},
30+
{firstName: "Michael", lastName: "Silva", position: "Marketing Designer", office: "London", salary: 198500},
31+
{firstName: "Timothy", lastName: "Mooney", position: "Office Manager", office: "London", salary: 136200},
32+
]
2433
/**
2534
* @title Table with sorting
2635
*/
@@ -29,11 +38,11 @@ const ELEMENT_DATA: PeriodicElement[] = [
2938
styleUrl: 'table-sorting-example.css',
3039
templateUrl: 'table-sorting-example.html',
3140
standalone: true,
32-
imports: [MatTableModule, MatSortModule],
41+
imports: [MatTableModule, MatSortModule, MatButtonToggle, MatButtonToggleGroup, MatButton],
3342
})
3443
export class TableSortingExample implements AfterViewInit {
35-
displayedColumns: string[] = ['position', 'name', 'weight', 'symbol'];
36-
dataSource = new MatTableDataSource(ELEMENT_DATA);
44+
displayedColumns: string[] = ['firstName', 'lastName', 'position', 'office', 'salary'];
45+
dataSource = new MatTableDataSource(MULTI_SORT_DATA);
3746

3847
constructor(private _liveAnnouncer: LiveAnnouncer) {}
3948

src/material/sort/sort-header.ts

+7-4
Original file line numberDiff line numberDiff line change
@@ -294,9 +294,10 @@ export class MatSortHeader implements MatSortable, OnDestroy, OnInit, AfterViewI
294294

295295
/** Whether this MatSortHeader is currently sorted in either ascending or descending order. */
296296
_isSorted() {
297+
const currentSortDirection = this._sort.getCurrentSortDirection(this.id);
297298
return (
298-
this._sort.active == this.id &&
299-
(this._sort.direction === 'asc' || this._sort.direction === 'desc')
299+
this._sort.isActive(this.id) &&
300+
(currentSortDirection === 'asc' || currentSortDirection === 'desc')
300301
);
301302
}
302303

@@ -322,7 +323,9 @@ export class MatSortHeader implements MatSortable, OnDestroy, OnInit, AfterViewI
322323
* only be changed once the arrow displays again (hint or activation).
323324
*/
324325
_updateArrowDirection() {
325-
this._arrowDirection = this._isSorted() ? this._sort.direction : this.start || this._sort.start;
326+
this._arrowDirection = this._isSorted()
327+
? this._sort.getCurrentSortDirection(this.id)
328+
: this.start || this._sort.start;
326329
}
327330

328331
_isDisabled() {
@@ -340,7 +343,7 @@ export class MatSortHeader implements MatSortable, OnDestroy, OnInit, AfterViewI
340343
return 'none';
341344
}
342345

343-
return this._sort.direction == 'asc' ? 'ascending' : 'descending';
346+
return this._sort.getCurrentSortDirection(this.id) == 'asc' ? 'ascending' : 'descending';
344347
}
345348

346349
/** Whether the arrow inside the sort header should be rendered. */

src/material/sort/sort.spec.ts

+21
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,9 @@ describe('MatSort', () => {
5757
fixture = TestBed.createComponent(SimpleMatSortApp);
5858
component = fixture.componentInstance;
5959
fixture.detectChanges();
60+
61+
component.matSort.matSortMultiple = false;
62+
component.matSort.sortState.clear();
6063
});
6164

6265
it('should have the sort headers register and deregister themselves', () => {
@@ -445,6 +448,24 @@ describe('MatSort', () => {
445448
expect(descriptionElement?.textContent).toBe('Sort 2nd column');
446449
});
447450

451+
it('should be able to store sorting for multiple columns when using multiSort', () => {
452+
component.matSort.matSortMultiple = true;
453+
454+
component.start = 'asc';
455+
testSingleColumnSortDirectionSequence(fixture, ['asc', 'desc', ''], 'defaultA');
456+
testSingleColumnSortDirectionSequence(fixture, ['asc', 'desc', ''], 'defaultB');
457+
458+
expect(component.matSort.sortState.size).toBe(2);
459+
460+
const defaultAState = component.matSort.sortState.get('defaultA');
461+
expect(defaultAState).toBeTruthy();
462+
expect(defaultAState?.direction).toBe(component.start);
463+
464+
const defaultBState = component.matSort.sortState.get('defaultB');
465+
expect(defaultBState).toBeTruthy();
466+
expect(defaultBState?.direction).toBe(component.start);
467+
});
468+
448469
it('should render arrows after sort header by default', () => {
449470
const matSortWithArrowPositionFixture = TestBed.createComponent(MatSortWithArrowPosition);
450471

src/material/sort/sort.ts

+79-6
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
*/
88

99
import {
10+
booleanAttribute,
1011
Directive,
1112
EventEmitter,
1213
Inject,
@@ -17,7 +18,7 @@ import {
1718
OnInit,
1819
Optional,
1920
Output,
20-
booleanAttribute,
21+
SimpleChanges,
2122
} from '@angular/core';
2223
import {Observable, ReplaySubject, Subject} from 'rxjs';
2324
import {SortDirection} from './sort-direction';
@@ -26,6 +27,7 @@ import {
2627
getSortHeaderMissingIdError,
2728
getSortInvalidDirectionError,
2829
} from './sort-errors';
30+
import {coerceBooleanProperty} from '@angular/cdk/coercion';
2931

3032
/** Position of the arrow that displays when sorted. */
3133
export type SortHeaderArrowPosition = 'before' | 'after';
@@ -79,6 +81,9 @@ export class MatSort implements OnChanges, OnDestroy, OnInit {
7981
/** Collection of all registered sortables that this directive manages. */
8082
sortables = new Map<string, MatSortable>();
8183

84+
/** Map holding the sort state for each column */
85+
sortState = new Map<string, Sort>;
86+
8287
/** Used to notify any child components listening to state changes. */
8388
readonly _stateChanges = new Subject<void>();
8489

@@ -109,6 +114,17 @@ export class MatSort implements OnChanges, OnDestroy, OnInit {
109114
}
110115
private _direction: SortDirection = '';
111116

117+
/** Whether to enable the multi-sorting feature */
118+
@Input('matSortMultiple')
119+
get matSortMultiple(): boolean {
120+
return this._sortMultiple;
121+
}
122+
123+
set matSortMultiple(value: any) {
124+
this._sortMultiple = coerceBooleanProperty(value);
125+
}
126+
private _sortMultiple = false;
127+
112128
/**
113129
* Whether to disable the user from clearing the sort by finishing the sort direction cycle.
114130
* May be overridden by the MatSortable's disable clear input.
@@ -160,14 +176,53 @@ export class MatSort implements OnChanges, OnDestroy, OnInit {
160176

161177
/** Sets the active sort id and determines the new sort direction. */
162178
sort(sortable: MatSortable): void {
179+
let sortableDirection;
180+
if (!this.isActive(sortable.id)) {
181+
sortableDirection = sortable.start ?? this.start;
182+
} else {
183+
sortableDirection = this.getNextSortDirection(sortable);
184+
}
185+
186+
// avoid keeping multiple sorts if not required.
187+
if (!this._sortMultiple) {
188+
this.sortState.clear();
189+
}
190+
191+
// Update active and direction to keep backwards compatibility
163192
if (this.active != sortable.id) {
164193
this.active = sortable.id;
165-
this.direction = sortable.start ? sortable.start : this.start;
194+
}
195+
this.direction = sortableDirection;
196+
197+
const currentSort: Sort = {
198+
active: sortable.id,
199+
direction: sortableDirection,
200+
};
201+
202+
// When unsorted, remove from state
203+
if (sortableDirection !== '') {
204+
this.sortState.set(sortable.id, currentSort);
166205
} else {
167-
this.direction = this.getNextSortDirection(sortable);
206+
this.sortState.delete(sortable.id);
168207
}
169208

170-
this.sortChange.emit({active: this.active, direction: this.direction});
209+
this.sortChange.emit(currentSort);
210+
}
211+
212+
/**
213+
* Checks whether the provided column is currently active (has been sorted)
214+
*/
215+
isActive(id: string): boolean {
216+
return this.sortState.has(id);
217+
}
218+
219+
/**
220+
* Returns the current SortDirection of the supplied column id, defaults to unsorted if no state is found.
221+
*/
222+
getCurrentSortDirection(id: string): SortDirection {
223+
return this.sortState.get(id)?.direction
224+
?? this.sortables.get(id)?.start
225+
?? this.start;
171226
}
172227

173228
/** Returns the next sort direction of the active sortable, checking for potential overrides. */
@@ -176,13 +231,14 @@ export class MatSort implements OnChanges, OnDestroy, OnInit {
176231
return '';
177232
}
178233

234+
const currentSortableDirection = this.getCurrentSortDirection(sortable.id);
179235
// Get the sort direction cycle with the potential sortable overrides.
180236
const disableClear =
181237
sortable?.disableClear ?? this.disableClear ?? !!this._defaultOptions?.disableClear;
182238
let sortDirectionCycle = getSortDirectionCycle(sortable.start || this.start, disableClear);
183239

184240
// Get and return the next direction in the cycle
185-
let nextDirectionIndex = sortDirectionCycle.indexOf(this.direction) + 1;
241+
let nextDirectionIndex = sortDirectionCycle.indexOf(currentSortableDirection) + 1;
186242
if (nextDirectionIndex >= sortDirectionCycle.length) {
187243
nextDirectionIndex = 0;
188244
}
@@ -193,7 +249,24 @@ export class MatSort implements OnChanges, OnDestroy, OnInit {
193249
this._initializedStream.next();
194250
}
195251

196-
ngOnChanges() {
252+
ngOnChanges(changes: SimpleChanges) {
253+
/* Update sortState with updated active and direction values, otherwise sorting won't work */
254+
if (changes['active'] || changes['direction']) {
255+
const currentActive = changes['active']?.currentValue ?? this.active;
256+
const currentDirection = changes['direction']?.currentValue ?? this.direction ?? this.start;
257+
258+
259+
// Handle sort deactivation
260+
if ((!currentActive || currentActive === '') && changes['active']?.previousValue) {
261+
this.sortState.delete(changes['active'].previousValue);
262+
} else {
263+
this.sortState.set(currentActive, {
264+
active: currentActive,
265+
direction: currentDirection,
266+
} as Sort);
267+
}
268+
}
269+
197270
this._stateChanges.next();
198271
}
199272

0 commit comments

Comments
 (0)