Skip to content

Commit 59ea92d

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 aff565a commit 59ea92d

File tree

4 files changed

+94
-6
lines changed

4 files changed

+94
-6
lines changed

src/lib/tabs/tab-body.ts

+48-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,22 +115,29 @@ 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;
121125

122126
/** Subscription to the directionality change observable. */
123127
private _dirChangeSubscription = Subscription.EMPTY;
124128

129+
/** Scroll position of the tab before the user switched away. */
130+
private _lastScrollPosition = 0;
131+
125132
/** Tab body position state. Used by the animation trigger for the current state. */
126133
_position: MatTabBodyPositionState;
127134

128135
/** Emits when an animation on the tab is complete. */
129136
_translateTabComplete = new Subject<AnimationEvent>();
130137

138+
/** Element wrapping the tab's content. */
139+
@ViewChild('content') _contentElement: ElementRef;
140+
131141
/** Event emitted when the tab begins to animate towards the center as the active tab. */
132142
@Output() readonly _onCentering: EventEmitter<number> = new EventEmitter<number>();
133143

@@ -154,6 +164,9 @@ export class MatTabBody implements OnInit, OnDestroy {
154164
/** Duration for the tab's animation. */
155165
@Input() animationDuration: string = '500ms';
156166

167+
/** Whether the tab is currently active. */
168+
@Input() active: boolean;
169+
157170
/** The shifted index position of the tab body, where zero represents the active center tab. */
158171
@Input()
159172
set position(position: number) {
@@ -214,16 +227,47 @@ export class MatTabBody implements OnInit, OnDestroy {
214227
}
215228
}
216229

230+
ngOnChanges(changes: SimpleChanges) {
231+
// Cache the scroll position before moving away from the tab. Note that this has to be done
232+
// through change detection and as early as possible, because some browsers (namely Safari)
233+
// will reset the scroll position when we switch from an absolute to a relative position.
234+
if (changes.active && changes.active.previousValue) {
235+
this._lastScrollPosition = this._elementRef.nativeElement.scrollTop ||
236+
this._contentElement.nativeElement.scrollTop;
237+
}
238+
}
239+
240+
_onTranslateTabComplete(e: AnimationEvent): void {
241+
// If the transition to the center is complete, emit an event.
242+
if (this._isCenterPosition(e.toState) && this._isCenterPosition(this._position)) {
243+
this._onCentered.emit();
244+
}
245+
246+
if (this._isCenterPosition(e.fromState) && !this._isCenterPosition(this._position)) {
247+
this._afterLeavingCenter.emit();
248+
}
249+
}
250+
217251
/** The text direction of the containing app. */
218252
_getLayoutDirection(): Direction {
219253
return this._dir && this._dir.value === 'rtl' ? 'rtl' : 'ltr';
220254
}
221255

222256
/** Whether the provided position state is considered center, regardless of origin. */
223-
_isCenterPosition(position: MatTabBodyPositionState|string): boolean {
257+
_isCenterPosition(position: MatTabBodyPositionState | string): boolean {
224258
return position == 'center' ||
225-
position == 'left-origin-center' ||
226-
position == 'right-origin-center';
259+
position == 'left-origin-center' ||
260+
position == 'right-origin-center';
261+
}
262+
263+
_restoreScrollPosition() {
264+
if (this._lastScrollPosition) {
265+
// Depending on the browser, the scrollable element can end up being
266+
// either the host element or the element with all the content.
267+
this._contentElement.nativeElement.scrollTop =
268+
this._elementRef.nativeElement.scrollTop =
269+
this._lastScrollPosition;
270+
}
227271
}
228272

229273
/** 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
@@ -478,6 +478,45 @@ describe('MatTabGroup', () => {
478478
expect(fixture.componentInstance.handleSelection).not.toHaveBeenCalled();
479479
}));
480480

481+
it('should preserve the scroll position when switching between tabs', fakeAsync(() => {
482+
const testComponent = fixture.componentInstance;
483+
484+
// Add a lot of content to make one of the tabs scrollable.
485+
testComponent.tabs[1].content = new Array(500).fill('content!').join('\n\n');
486+
fixture.detectChanges();
487+
tick(500);
488+
489+
// Cap the tab group height.
490+
fixture.debugElement.query(By.css('mat-tab-group')).nativeElement.style.height = `300px`;
491+
492+
const tabElements = fixture.debugElement.queryAll(By.css('.mat-tab-body-content'))
493+
.map(debugElement => debugElement.nativeElement as HTMLElement);
494+
495+
// Focus the tab with the extra content.
496+
testComponent.selectedIndex = 1;
497+
fixture.detectChanges();
498+
tick(500);
499+
500+
// Ensure that there is content and scroll down 100px.
501+
expect(tabElements[1].offsetHeight).toBeGreaterThan(0, 'Expected tab to have some content.');
502+
503+
// Handle some differences in the way browsers determine what element is scrollable.
504+
tabElements[1].scrollTop = tabElements[1].parentElement!.scrollTop = 100;
505+
506+
// Move to another tab.
507+
testComponent.selectedIndex = 0;
508+
fixture.detectChanges();
509+
tick(500);
510+
511+
// Switch back to the tab with the extra content.
512+
testComponent.selectedIndex = 1;
513+
fixture.detectChanges();
514+
tick(500);
515+
516+
expect(tabElements[1].scrollTop || tabElements[1].parentElement!.scrollTop)
517+
.toBe(100, 'Expected scroll position to be restored.');
518+
}));
519+
481520
});
482521

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

tools/public_api_guard/lib/tabs.d.ts

+6-1
Original file line numberDiff line numberDiff line change
@@ -36,23 +36,28 @@ export declare class MatTab extends _MatTabMixinBase implements OnInit, CanDisab
3636
ngOnInit(): void;
3737
}
3838

39-
export declare class MatTabBody implements OnInit, OnDestroy {
39+
export declare class MatTabBody implements OnInit, OnChanges, OnDestroy {
4040
readonly _afterLeavingCenter: EventEmitter<boolean>;
4141
readonly _beforeCentering: EventEmitter<boolean>;
4242
_content: TemplatePortal;
43+
_contentElement: ElementRef;
4344
readonly _onCentered: EventEmitter<void>;
4445
readonly _onCentering: EventEmitter<number>;
4546
_portalHost: PortalHostDirective;
4647
_position: MatTabBodyPositionState;
4748
_translateTabComplete: Subject<AnimationEvent>;
49+
active: boolean;
4850
animationDuration: string;
4951
origin: number;
5052
position: number;
5153
constructor(_elementRef: ElementRef<HTMLElement>, _dir: Directionality,
5254
changeDetectorRef?: ChangeDetectorRef);
5355
_getLayoutDirection(): Direction;
5456
_isCenterPosition(position: MatTabBodyPositionState | string): boolean;
57+
_onTranslateTabComplete(e: AnimationEvent): void;
5558
_onTranslateTabStarted(event: AnimationEvent): void;
59+
_restoreScrollPosition(): void;
60+
ngOnChanges(changes: SimpleChanges): void;
5661
ngOnDestroy(): void;
5762
ngOnInit(): void;
5863
}

0 commit comments

Comments
 (0)