Skip to content

Commit 978ee67

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 dabb967 commit 978ee67

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
styleUrls: ['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 {HasInitialized, mixinInitialized} from '@angular/material/core';
2324
import {Subject} from 'rxjs';
@@ -27,6 +28,7 @@ import {
2728
getSortHeaderMissingIdError,
2829
getSortInvalidDirectionError,
2930
} from './sort-errors';
31+
import {coerceBooleanProperty} from '@angular/cdk/coercion';
3032

3133
/** Position of the arrow that displays when sorted. */
3234
export type SortHeaderArrowPosition = 'before' | 'after';
@@ -82,6 +84,9 @@ export class MatSort extends _MatSortBase implements HasInitialized, OnChanges,
8284
/** Collection of all registered sortables that this directive manages. */
8385
sortables = new Map<string, MatSortable>();
8486

87+
/** Map holding the sort state for each column */
88+
sortState = new Map<string, Sort>;
89+
8590
/** Used to notify any child components listening to state changes. */
8691
readonly _stateChanges = new Subject<void>();
8792

@@ -112,6 +117,17 @@ export class MatSort extends _MatSortBase implements HasInitialized, OnChanges,
112117
}
113118
private _direction: SortDirection = '';
114119

120+
/** Whether to enable the multi-sorting feature */
121+
@Input('matSortMultiple')
122+
get matSortMultiple(): boolean {
123+
return this._sortMultiple;
124+
}
125+
126+
set matSortMultiple(value: any) {
127+
this._sortMultiple = coerceBooleanProperty(value);
128+
}
129+
private _sortMultiple = false;
130+
115131
/**
116132
* Whether to disable the user from clearing the sort by finishing the sort direction cycle.
117133
* May be overridden by the MatSortable's disable clear input.
@@ -162,14 +178,53 @@ export class MatSort extends _MatSortBase implements HasInitialized, OnChanges,
162178

163179
/** Sets the active sort id and determines the new sort direction. */
164180
sort(sortable: MatSortable): void {
181+
let sortableDirection;
182+
if (!this.isActive(sortable.id)) {
183+
sortableDirection = sortable.start ?? this.start;
184+
} else {
185+
sortableDirection = this.getNextSortDirection(sortable);
186+
}
187+
188+
// avoid keeping multiple sorts if not required.
189+
if (!this._sortMultiple) {
190+
this.sortState.clear();
191+
}
192+
193+
// Update active and direction to keep backwards compatibility
165194
if (this.active != sortable.id) {
166195
this.active = sortable.id;
167-
this.direction = sortable.start ? sortable.start : this.start;
196+
}
197+
this.direction = sortableDirection;
198+
199+
const currentSort: Sort = {
200+
active: sortable.id,
201+
direction: sortableDirection,
202+
};
203+
204+
// When unsorted, remove from state
205+
if (sortableDirection !== '') {
206+
this.sortState.set(sortable.id, currentSort);
168207
} else {
169-
this.direction = this.getNextSortDirection(sortable);
208+
this.sortState.delete(sortable.id);
170209
}
171210

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

175230
/** Returns the next sort direction of the active sortable, checking for potential overrides. */
@@ -178,13 +233,14 @@ export class MatSort extends _MatSortBase implements HasInitialized, OnChanges,
178233
return '';
179234
}
180235

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

186242
// Get and return the next direction in the cycle
187-
let nextDirectionIndex = sortDirectionCycle.indexOf(this.direction) + 1;
243+
let nextDirectionIndex = sortDirectionCycle.indexOf(currentSortableDirection) + 1;
188244
if (nextDirectionIndex >= sortDirectionCycle.length) {
189245
nextDirectionIndex = 0;
190246
}
@@ -195,7 +251,24 @@ export class MatSort extends _MatSortBase implements HasInitialized, OnChanges,
195251
this._markInitialized();
196252
}
197253

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

0 commit comments

Comments
 (0)