Skip to content

Commit afc4f88

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 f9b9e25 commit afc4f88

File tree

3 files changed

+77
-5
lines changed

3 files changed

+77
-5
lines changed

src/lib/tabs/tab-body.ts

+37-4
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,8 @@ import {
2424
ViewContainerRef,
2525
forwardRef,
2626
ViewChild,
27+
OnChanges,
28+
SimpleChanges,
2729
} from '@angular/core';
2830
import {AnimationEvent} from '@angular/animations';
2931
import {TemplatePortal, CdkPortalOutlet, PortalHostDirective} from '@angular/cdk/portal';
@@ -82,6 +84,7 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
8284
.subscribe((isCentering: boolean) => {
8385
if (isCentering && !this.hasAttached()) {
8486
this.attach(this._host._content);
87+
this._host._restoreScrollPosition();
8588
}
8689
});
8790

@@ -112,9 +115,10 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
112115
animations: [matTabsAnimations.translateTab],
113116
host: {
114117
'class': 'mat-tab-body',
118+
'[class.mat-tab-body-active]': 'active',
115119
},
116120
})
117-
export class MatTabBody implements OnInit, OnDestroy {
121+
export class MatTabBody implements OnInit, OnChanges, OnDestroy {
118122

119123
/** Current position of the tab-body in the tab-group. Zero means that the tab is visible. */
120124
private _positionIndex: number;
@@ -125,6 +129,9 @@ export class MatTabBody implements OnInit, OnDestroy {
125129
/** Tab body position state. Used by the animation trigger for the current state. */
126130
_position: MatTabBodyPositionState;
127131

132+
/** Element wrapping the tab's content. */
133+
@ViewChild('content') _contentElement: ElementRef;
134+
128135
/** Event emitted when the tab begins to animate towards the center as the active tab. */
129136
@Output() readonly _onCentering: EventEmitter<number> = new EventEmitter<number>();
130137

@@ -146,6 +153,12 @@ export class MatTabBody implements OnInit, OnDestroy {
146153
/** Position that will be used when the tab is immediately becoming visible after creation. */
147154
@Input() origin: number;
148155

156+
/** Whether the tab is currently active. */
157+
@Input() active: boolean;
158+
159+
/** Scroll position of the tab before the user switched away. */
160+
private _lastScrollPosition = 0;
161+
149162
/** The shifted index position of the tab body, where zero represents the active center tab. */
150163
@Input()
151164
set position(position: number) {
@@ -190,6 +203,16 @@ export class MatTabBody implements OnInit, OnDestroy {
190203
}
191204
}
192205

206+
ngOnChanges(changes: SimpleChanges) {
207+
// Cache the scroll position before moving away from the tab. Note that this has to be done
208+
// through change detection and as early as possible, because some browsers (namely Safari)
209+
// will reset the scroll position when we switch from an absolute to a relative position.
210+
if (changes.active && changes.active.previousValue) {
211+
this._lastScrollPosition = this._elementRef.nativeElement.scrollTop ||
212+
this._contentElement.nativeElement.scrollTop;
213+
}
214+
}
215+
193216
_onTranslateTabComplete(e: AnimationEvent): void {
194217
// If the transition to the center is complete, emit an event.
195218
if (this._isCenterPosition(e.toState) && this._isCenterPosition(this._position)) {
@@ -207,10 +230,20 @@ export class MatTabBody implements OnInit, OnDestroy {
207230
}
208231

209232
/** Whether the provided position state is considered center, regardless of origin. */
210-
_isCenterPosition(position: MatTabBodyPositionState|string): boolean {
233+
_isCenterPosition(position: MatTabBodyPositionState | string): boolean {
211234
return position == 'center' ||
212-
position == 'left-origin-center' ||
213-
position == 'right-origin-center';
235+
position == 'left-origin-center' ||
236+
position == 'right-origin-center';
237+
}
238+
239+
_restoreScrollPosition() {
240+
if (this._lastScrollPosition) {
241+
// Depending on the browser, the scrollable element can end up being
242+
// either the host element or the element with all the content.
243+
this._contentElement.nativeElement.scrollTop =
244+
this._elementRef.nativeElement.scrollTop =
245+
this._lastScrollPosition;
246+
}
214247
}
215248

216249
/** Computes the position state that will be used for the tab-body animation trigger. */

src/lib/tabs/tab-group.html

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

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

+39
Original file line numberDiff line numberDiff line change
@@ -463,6 +463,45 @@ describe('MatTabGroup', () => {
463463
expect(fixture.componentInstance.handleSelection).not.toHaveBeenCalled();
464464
}));
465465

466+
it('should preserve the scroll position when switching between tabs', fakeAsync(() => {
467+
const testComponent = fixture.componentInstance;
468+
469+
// Add a lot of content to make one of the tabs scrollable.
470+
testComponent.tabs[1].content = new Array(500).fill('content!').join('\n\n');
471+
fixture.detectChanges();
472+
tick(500);
473+
474+
// Cap the tab group height.
475+
fixture.debugElement.query(By.css('mat-tab-group')).nativeElement.style.height = `300px`;
476+
477+
const tabElements = fixture.debugElement.queryAll(By.css('.mat-tab-body-content'))
478+
.map(debugElement => debugElement.nativeElement as HTMLElement);
479+
480+
// Focus the tab with the extra content.
481+
testComponent.selectedIndex = 1;
482+
fixture.detectChanges();
483+
tick(500);
484+
485+
// Ensure that there is content and scroll down 100px.
486+
expect(tabElements[1].offsetHeight).toBeGreaterThan(0, 'Expected tab to have some content.');
487+
488+
// Handle some differences in the way browsers determine what element is scrollable.
489+
tabElements[1].scrollTop = tabElements[1].parentElement!.scrollTop = 100;
490+
491+
// Move to another tab.
492+
testComponent.selectedIndex = 0;
493+
fixture.detectChanges();
494+
tick(500);
495+
496+
// Switch back to the tab with the extra content.
497+
testComponent.selectedIndex = 1;
498+
fixture.detectChanges();
499+
tick(500);
500+
501+
expect(tabElements[1].scrollTop || tabElements[1].parentElement!.scrollTop)
502+
.toBe(100, 'Expected scroll position to be restored.');
503+
}));
504+
466505
});
467506

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

0 commit comments

Comments
 (0)