Skip to content

Commit c16f8e3

Browse files
committed
fix(tabs): preserve scroll position when switching between tabs
Preserves the scroll position when switching between tabs. Previously it was being reset to 0, because we detach and re-attach the content. Fixes #6722.
1 parent 7857b92 commit c16f8e3

File tree

3 files changed

+77
-5
lines changed

3 files changed

+77
-5
lines changed

src/lib/tabs/tab-body.ts

+38-4
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@ import {
2323
ViewContainerRef,
2424
forwardRef,
2525
ViewChild,
26+
OnChanges,
27+
SimpleChanges,
2628
} from '@angular/core';
2729
import {AnimationEvent} from '@angular/animations';
2830
import {TemplatePortal, CdkPortalOutlet, PortalHostDirective} from '@angular/cdk/portal';
@@ -75,10 +77,12 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
7577
ngOnInit(): void {
7678
if (this._host._isCenterPosition(this._host._position)) {
7779
this.attach(this._host._content);
80+
this._host._restoreScrollPosition();
7881
}
7982
this._centeringSub = this._host._beforeCentering.subscribe((isCentering: boolean) => {
8083
if (isCentering && !this.hasAttached()) {
8184
this.attach(this._host._content);
85+
this._host._restoreScrollPosition();
8286
}
8387
});
8488

@@ -113,9 +117,13 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
113117
animations: [matTabsAnimations.translateTab],
114118
host: {
115119
'class': 'mat-tab-body',
120+
'[class.mat-tab-body-active]': 'active',
116121
},
117122
})
118-
export class MatTabBody implements OnInit {
123+
export class MatTabBody implements OnInit, OnChanges {
124+
/** Element wrapping the tab's content. */
125+
@ViewChild('content') _contentElement: ElementRef;
126+
119127
/** Event emitted when the tab begins to animate towards the center as the active tab. */
120128
@Output() readonly _onCentering: EventEmitter<number> = new EventEmitter<number>();
121129

@@ -134,6 +142,12 @@ export class MatTabBody implements OnInit {
134142
/** The tab body content to display. */
135143
@Input('content') _content: TemplatePortal;
136144

145+
/** Whether the tab is currently active. */
146+
@Input() active: boolean;
147+
148+
/** Scroll position of the tab before the user switched away. */
149+
private _lastScrollPosition = 0;
150+
137151
/** The shifted index position of the tab body, where zero represents the active center tab. */
138152
@Input()
139153
set position(position: number) {
@@ -182,6 +196,16 @@ export class MatTabBody implements OnInit {
182196
}
183197
}
184198

199+
ngOnChanges(changes: SimpleChanges) {
200+
// Cache the scroll position before moving away from the tab. Note that this has to be done
201+
// through change detection and as early as possible, because some browsers (namely Safari)
202+
// will reset the scroll position when we switch from an absolute to a relative position.
203+
if (changes.active && changes.active.previousValue) {
204+
this._lastScrollPosition = this._elementRef.nativeElement.scrollTop ||
205+
this._contentElement.nativeElement.scrollTop;
206+
}
207+
}
208+
185209
_onTranslateTabComplete(e: AnimationEvent): void {
186210
// If the transition to the center is complete, emit an event.
187211
if (this._isCenterPosition(e.toState) && this._isCenterPosition(this._position)) {
@@ -199,9 +223,19 @@ export class MatTabBody implements OnInit {
199223
}
200224

201225
/** Whether the provided position state is considered center, regardless of origin. */
202-
_isCenterPosition(position: MatTabBodyPositionState|string): boolean {
226+
_isCenterPosition(position: MatTabBodyPositionState | string): boolean {
203227
return position == 'center' ||
204-
position == 'left-origin-center' ||
205-
position == 'right-origin-center';
228+
position == 'left-origin-center' ||
229+
position == 'right-origin-center';
230+
}
231+
232+
_restoreScrollPosition() {
233+
if (this._lastScrollPosition) {
234+
// Depending on the browser, the scrollable element can end up being
235+
// either the host element or the element with all the content.
236+
this._contentElement.nativeElement.scrollTop =
237+
this._elementRef.nativeElement.scrollTop =
238+
this._lastScrollPosition;
239+
}
206240
}
207241
}

src/lib/tabs/tab-group.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@
3232
*ngFor="let tab of _tabs; let i = index"
3333
[id]="_getTabContentId(i)"
3434
[attr.aria-labelledby]="_getTabLabelId(i)"
35-
[class.mat-tab-body-active]="selectedIndex == i"
35+
[active]="selectedIndex === i"
3636
[content]="tab.content"
3737
[position]="tab.position"
3838
[origin]="tab.origin"

src/lib/tabs/tab-group.spec.ts

+38
Original file line numberDiff line numberDiff line change
@@ -309,6 +309,44 @@ describe('MatTabGroup', () => {
309309
expect(component.selectedIndex).toBe(numberOfTabs - 2);
310310
}));
311311

312+
it('should preserve the scroll position when switching between tabs', fakeAsync(() => {
313+
const testComponent = fixture.componentInstance;
314+
315+
// Add a lot of content to make one of the tabs scrollable.
316+
testComponent.tabs[1].content = new Array(500).fill('content!').join('\n\n');
317+
fixture.detectChanges();
318+
tick(500);
319+
320+
// Cap the tab group height.
321+
fixture.debugElement.query(By.css('mat-tab-group')).nativeElement.style.height = `300px`;
322+
323+
const tabElements = fixture.debugElement.queryAll(By.css('.mat-tab-body-content'))
324+
.map(debugElement => debugElement.nativeElement as HTMLElement);
325+
326+
// Focus the tab with the extra content.
327+
testComponent.selectedIndex = 1;
328+
fixture.detectChanges();
329+
tick(500);
330+
331+
// Ensure that there is content and scroll down 100px.
332+
expect(tabElements[1].offsetHeight).toBeGreaterThan(0, 'Expected tab to have some content.');
333+
334+
// Handle some differences in the way browsers determine what element is scrollable.
335+
tabElements[1].scrollTop = tabElements[1].parentElement!.scrollTop = 100;
336+
337+
// Move to another tab.
338+
testComponent.selectedIndex = 0;
339+
fixture.detectChanges();
340+
tick(500);
341+
342+
// Switch back to the tab with the extra content.
343+
testComponent.selectedIndex = 1;
344+
fixture.detectChanges();
345+
tick(500);
346+
347+
expect(tabElements[1].scrollTop || tabElements[1].parentElement!.scrollTop)
348+
.toBe(100, 'Expected scroll position to be restored.');
349+
}));
312350
});
313351

314352
describe('async tabs', () => {

0 commit comments

Comments
 (0)