@@ -24,6 +24,8 @@ import {
24
24
ViewContainerRef ,
25
25
forwardRef ,
26
26
ViewChild ,
27
+ OnChanges ,
28
+ SimpleChanges ,
27
29
} from '@angular/core' ;
28
30
import { AnimationEvent } from '@angular/animations' ;
29
31
import { TemplatePortal , CdkPortalOutlet , PortalHostDirective } from '@angular/cdk/portal' ;
@@ -82,6 +84,7 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
82
84
. subscribe ( ( isCentering : boolean ) => {
83
85
if ( isCentering && ! this . hasAttached ( ) ) {
84
86
this . attach ( this . _host . _content ) ;
87
+ this . _host . _restoreScrollPosition ( ) ;
85
88
}
86
89
} ) ;
87
90
@@ -112,9 +115,10 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
112
115
animations : [ matTabsAnimations . translateTab ] ,
113
116
host : {
114
117
'class' : 'mat-tab-body' ,
118
+ '[class.mat-tab-body-active]' : 'active' ,
115
119
} ,
116
120
} )
117
- export class MatTabBody implements OnInit , OnDestroy {
121
+ export class MatTabBody implements OnInit , OnChanges , OnDestroy {
118
122
119
123
/** Current position of the tab-body in the tab-group. Zero means that the tab is visible. */
120
124
private _positionIndex : number ;
@@ -125,6 +129,9 @@ export class MatTabBody implements OnInit, OnDestroy {
125
129
/** Tab body position state. Used by the animation trigger for the current state. */
126
130
_position : MatTabBodyPositionState ;
127
131
132
+ /** Element wrapping the tab's content. */
133
+ @ViewChild ( 'content' ) _contentElement : ElementRef ;
134
+
128
135
/** Event emitted when the tab begins to animate towards the center as the active tab. */
129
136
@Output ( ) readonly _onCentering : EventEmitter < number > = new EventEmitter < number > ( ) ;
130
137
@@ -146,6 +153,12 @@ export class MatTabBody implements OnInit, OnDestroy {
146
153
/** Position that will be used when the tab is immediately becoming visible after creation. */
147
154
@Input ( ) origin : number ;
148
155
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
+
149
162
/** The shifted index position of the tab body, where zero represents the active center tab. */
150
163
@Input ( )
151
164
set position ( position : number ) {
@@ -190,6 +203,16 @@ export class MatTabBody implements OnInit, OnDestroy {
190
203
}
191
204
}
192
205
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
+
193
216
_onTranslateTabComplete ( e : AnimationEvent ) : void {
194
217
// If the transition to the center is complete, emit an event.
195
218
if ( this . _isCenterPosition ( e . toState ) && this . _isCenterPosition ( this . _position ) ) {
@@ -207,10 +230,20 @@ export class MatTabBody implements OnInit, OnDestroy {
207
230
}
208
231
209
232
/** Whether the provided position state is considered center, regardless of origin. */
210
- _isCenterPosition ( position : MatTabBodyPositionState | string ) : boolean {
233
+ _isCenterPosition ( position : MatTabBodyPositionState | string ) : boolean {
211
234
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
+ }
214
247
}
215
248
216
249
/** Computes the position state that will be used for the tab-body animation trigger. */
0 commit comments