Skip to content

Commit 75c8aa8

Browse files
authoredDec 10, 2024··
feat(cdk-experimental/column-resize): Support column size persistance hooks (#30136)
1 parent 4a0818d commit 75c8aa8

File tree

5 files changed

+142
-11
lines changed

5 files changed

+142
-11
lines changed
 

‎src/cdk-experimental/column-resize/column-resize.ts

+6-2
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,6 @@ export const COLUMN_RESIZE_OPTIONS = new InjectionToken<ColumnResizeOptions>(
4444
*/
4545
@Directive()
4646
export abstract class ColumnResize implements AfterViewInit, OnDestroy {
47-
private _idGenerator = inject(_IdGenerator);
4847
protected readonly destroyed = new Subject<void>();
4948

5049
/* Publicly accessible interface for triggering and being notified of resizes. */
@@ -58,7 +57,7 @@ export abstract class ColumnResize implements AfterViewInit, OnDestroy {
5857
protected abstract readonly notifier: ColumnResizeNotifierSource;
5958

6059
/** Unique ID for this table instance. */
61-
protected readonly selectorId = this._idGenerator.getId('cdk-column-resize-');
60+
protected readonly selectorId = inject(_IdGenerator).getId('cdk-column-resize-');
6261

6362
/** The id attribute of the table, if specified. */
6463
id?: string;
@@ -88,6 +87,11 @@ export abstract class ColumnResize implements AfterViewInit, OnDestroy {
8887
return this.selectorId;
8988
}
9089

90+
/** Gets the ID for this table used for column size persistance. */
91+
getTableId(): string {
92+
return String(this.elementRef.nativeElement.id);
93+
}
94+
9195
/** Called when a column in the table is resized. Applies a css class to the table element. */
9296
setResized() {
9397
this.elementRef.nativeElement!.classList.add(WITH_RESIZED_COLUMN_CLASS);

‎src/cdk-experimental/column-resize/column-size-store.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -7,15 +7,16 @@
77
*/
88

99
import {Injectable} from '@angular/core';
10+
import {Observable} from 'rxjs';
1011

1112
/**
1213
* Can be provided by the host application to enable persistence of column resize state.
1314
*/
1415
@Injectable()
1516
export abstract class ColumnSizeStore {
1617
/** Returns the persisted size of the specified column in the specified table. */
17-
abstract getSize(tableId: string, columnId: string): number;
18+
abstract getSize(tableId: string, columnId: string): Observable<number | null> | null;
1819

1920
/** Persists the size of the specified column in the specified table. */
20-
abstract setSize(tableId: string, columnId: string): void;
21+
abstract setSize(tableId: string, columnId: string, sizePx: number): void;
2122
}

‎src/cdk-experimental/column-resize/resizable.ts

+28-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import {
1010
AfterViewInit,
1111
Directive,
1212
ElementRef,
13+
inject,
1314
Injector,
1415
NgZone,
1516
OnDestroy,
@@ -22,14 +23,15 @@ import {ComponentPortal} from '@angular/cdk/portal';
2223
import {Overlay, OverlayRef} from '@angular/cdk/overlay';
2324
import {CdkColumnDef, _CoalescedStyleScheduler} from '@angular/cdk/table';
2425
import {merge, Subject} from 'rxjs';
25-
import {filter, takeUntil} from 'rxjs/operators';
26+
import {distinctUntilChanged, filter, take, takeUntil} from 'rxjs/operators';
2627

2728
import {_closest} from '@angular/cdk-experimental/popover-edit';
2829

2930
import {HEADER_ROW_SELECTOR} from './selectors';
3031
import {ResizeOverlayHandle} from './overlay-handle';
3132
import {ColumnResize} from './column-resize';
3233
import {ColumnSizeAction, ColumnResizeNotifierSource} from './column-resize-notifier';
34+
import {ColumnSizeStore} from './column-size-store';
3335
import {HeaderRowEventDispatcher} from './event-dispatcher';
3436
import {ResizeRef} from './resize-ref';
3537
import {ResizeStrategy} from './resize-strategy';
@@ -66,6 +68,8 @@ export abstract class Resizable<HandleComponent extends ResizeOverlayHandle>
6668
protected abstract readonly viewContainerRef: ViewContainerRef;
6769
protected abstract readonly changeDetectorRef: ChangeDetectorRef;
6870

71+
protected readonly columnSizeStore = inject(ColumnSizeStore, {optional: true});
72+
6973
private _viewInitialized = false;
7074
private _isDestroyed = false;
7175

@@ -105,6 +109,15 @@ export abstract class Resizable<HandleComponent extends ResizeOverlayHandle>
105109
this._viewInitialized = true;
106110
this._applyMinWidthPx();
107111
this._applyMaxWidthPx();
112+
this.columnSizeStore
113+
?.getSize(this.columnResize.getTableId(), this.columnDef.name)
114+
?.pipe(take(1), takeUntil(this.destroyed))
115+
.subscribe(size => {
116+
if (size == null) {
117+
return;
118+
}
119+
this._applySize(size);
120+
});
108121
});
109122
}
110123

@@ -195,6 +208,20 @@ export abstract class Resizable<HandleComponent extends ResizeOverlayHandle>
195208
.subscribe(columnSize => {
196209
this._cleanUpAfterResize(columnSize);
197210
});
211+
212+
this.resizeNotifier.resizeCompleted
213+
.pipe(
214+
filter(sizeUpdate => sizeUpdate.columnId === this.columnDef.name),
215+
distinctUntilChanged((a, b) => a.size === b.size),
216+
takeUntil(this.destroyed),
217+
)
218+
.subscribe(sizeUpdate => {
219+
this.columnSizeStore?.setSize(
220+
this.columnResize.getTableId(),
221+
this.columnDef.name,
222+
sizeUpdate.size,
223+
);
224+
});
198225
}
199226

200227
private _completeResizeOperation(): void {

‎src/material-experimental/column-resize/column-resize.spec.ts

+104-5
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,20 @@
11
import {BidiModule} from '@angular/cdk/bidi';
22
import {DataSource} from '@angular/cdk/collections';
33
import {ESCAPE} from '@angular/cdk/keycodes';
4-
import {ChangeDetectionStrategy, Component, Directive, ElementRef, ViewChild} from '@angular/core';
4+
import {
5+
ChangeDetectionStrategy,
6+
Component,
7+
Directive,
8+
ElementRef,
9+
Injectable,
10+
ViewChild,
11+
} from '@angular/core';
512
import {ComponentFixture, TestBed, fakeAsync, flush} from '@angular/core/testing';
613
import {MatTableModule} from '@angular/material/table';
7-
import {BehaviorSubject} from 'rxjs';
14+
import {BehaviorSubject, Observable, ReplaySubject} from 'rxjs';
815
import {dispatchKeyboardEvent} from '../../cdk/testing/private';
916

10-
import {ColumnSize} from '@angular/cdk-experimental/column-resize';
17+
import {ColumnSize, ColumnSizeStore} from '@angular/cdk-experimental/column-resize';
1118
import {AbstractMatColumnResize} from './column-resize-directives/common';
1219
import {
1320
MatColumnResize,
@@ -52,7 +59,7 @@ function getTableTemplate(defaultEnabled: boolean) {
5259
}
5360
</style>
5461
<div #table [dir]="direction">
55-
<table ${directives.table} mat-table [dataSource]="dataSource"
62+
<table ${directives.table} mat-table [dataSource]="dataSource" id="theTable"
5663
style="table-layout: fixed;">
5764
<!-- Position Column -->
5865
<ng-container matColumnDef="position" sticky>
@@ -109,7 +116,7 @@ function getFlexTemplate(defaultEnabled: boolean) {
109116
}
110117
</style>
111118
<div #table [dir]="direction">
112-
<mat-table ${directives.table} [dataSource]="dataSource">
119+
<mat-table ${directives.table} [dataSource]="dataSource" id="theTable">
113120
<!-- Position Column -->
114121
<ng-container matColumnDef="position" sticky>
115122
<mat-header-cell *matHeaderCellDef
@@ -628,6 +635,63 @@ describe('Material Popover Edit', () => {
628635
}));
629636
});
630637
}
638+
639+
describe('ColumnSizeStore (persistance)', () => {
640+
let component: BaseTestComponent;
641+
let fixture: ComponentFixture<BaseTestComponent>;
642+
let columnSizeStore: FakeColumnSizeStore;
643+
644+
beforeEach(fakeAsync(() => {
645+
jasmine.addMatchers(approximateMatcher);
646+
647+
TestBed.configureTestingModule({
648+
imports: [BidiModule, MatTableModule, MatColumnResizeModule],
649+
providers: [
650+
FakeColumnSizeStore,
651+
{provide: ColumnSizeStore, useExisting: FakeColumnSizeStore},
652+
],
653+
declarations: [MatResizeOnPushTest],
654+
});
655+
fixture = TestBed.createComponent(MatResizeOnPushTest);
656+
component = fixture.componentInstance;
657+
columnSizeStore = TestBed.inject(FakeColumnSizeStore);
658+
fixture.detectChanges();
659+
flush();
660+
}));
661+
662+
it('applies the persisted size', fakeAsync(() => {
663+
(expect(component.getColumnWidth(1)).not as any).isApproximately(300);
664+
665+
columnSizeStore.emitSize('theTable', 'name', 300);
666+
667+
flush();
668+
669+
(expect(component.getColumnWidth(1)) as any).isApproximately(300);
670+
}));
671+
672+
it('persists the user-triggered size update', fakeAsync(() => {
673+
const initialColumnWidth = component.getColumnWidth(1);
674+
675+
component.triggerHoverState();
676+
fixture.detectChanges();
677+
678+
component.resizeColumnWithMouse(1, 5);
679+
fixture.detectChanges();
680+
flush();
681+
682+
component.completeResizeWithMouseInProgress(1);
683+
flush();
684+
685+
component.endHoverState();
686+
fixture.detectChanges();
687+
688+
expect(columnSizeStore.setSizeCalls.length).toBe(1);
689+
const {tableId, columnId, sizePx} = columnSizeStore.setSizeCalls[0];
690+
expect(tableId).toBe('theTable');
691+
expect(columnId).toBe('name');
692+
(expect(sizePx) as any).isApproximately(initialColumnWidth + 5);
693+
}));
694+
});
631695
});
632696

633697
function createElementData() {
@@ -639,3 +703,38 @@ function createElementData() {
639703
{position: 5, name: 'Boron', weight: 10.811, symbol: 'B'},
640704
];
641705
}
706+
707+
@Injectable()
708+
class FakeColumnSizeStore extends ColumnSizeStore {
709+
readonly emitStore = new Map<string, ReplaySubject<number>>();
710+
readonly setSizeCalls: {tableId: string; columnId: string; sizePx: number}[] = [];
711+
712+
/** Returns an observable that will emit values from emitSize(). */
713+
override getSize(tableId: string, columnId: string): Observable<number> | null {
714+
return this._getOrAdd(tableId, columnId);
715+
}
716+
717+
/**
718+
* Adds an entry to setSizeCalls.
719+
* Note: Does not affect values returned from getSize.
720+
*/
721+
override setSize(tableId: string, columnId: string, sizePx: number): void {
722+
this.setSizeCalls.push({tableId, columnId, sizePx});
723+
}
724+
725+
/** Call this in test code to simulate persisted column sizes. */
726+
emitSize(tableId: string, columnId: string, sizePx: number) {
727+
const stored = this._getOrAdd(tableId, columnId);
728+
stored.next(sizePx);
729+
}
730+
731+
private _getOrAdd(tableId: string, columnId: string): ReplaySubject<number> {
732+
const key = `tableId----columnId`;
733+
let stored = this.emitStore.get(key);
734+
if (!stored) {
735+
stored = new ReplaySubject<number>(1);
736+
this.emitStore.set(key, stored);
737+
}
738+
return stored;
739+
}
740+
}

‎src/material-experimental/column-resize/public-api.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,4 +16,4 @@ export * from './resizable-directives/resizable';
1616
export * from './resize-strategy';
1717
export * from './overlay-handle';
1818
export type {ColumnResizeOptions} from '@angular/cdk-experimental/column-resize';
19-
export {COLUMN_RESIZE_OPTIONS} from '@angular/cdk-experimental/column-resize';
19+
export {COLUMN_RESIZE_OPTIONS, ColumnSizeStore} from '@angular/cdk-experimental/column-resize';

0 commit comments

Comments
 (0)
Please sign in to comment.