@@ -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,22 +115,29 @@ 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 ;
121
125
122
126
/** Subscription to the directionality change observable. */
123
127
private _dirChangeSubscription = Subscription . EMPTY ;
124
128
129
+ /** Scroll position of the tab before the user switched away. */
130
+ private _lastScrollPosition = 0 ;
131
+
125
132
/** Tab body position state. Used by the animation trigger for the current state. */
126
133
_position : MatTabBodyPositionState ;
127
134
128
135
/** Emits when an animation on the tab is complete. */
129
136
_translateTabComplete = new Subject < AnimationEvent > ( ) ;
130
137
138
+ /** Element wrapping the tab's content. */
139
+ @ViewChild ( 'content' ) _contentElement : ElementRef ;
140
+
131
141
/** Event emitted when the tab begins to animate towards the center as the active tab. */
132
142
@Output ( ) readonly _onCentering : EventEmitter < number > = new EventEmitter < number > ( ) ;
133
143
@@ -154,6 +164,9 @@ export class MatTabBody implements OnInit, OnDestroy {
154
164
/** Duration for the tab's animation. */
155
165
@Input ( ) animationDuration : string = '500ms' ;
156
166
167
+ /** Whether the tab is currently active. */
168
+ @Input ( ) active : boolean ;
169
+
157
170
/** The shifted index position of the tab body, where zero represents the active center tab. */
158
171
@Input ( )
159
172
set position ( position : number ) {
@@ -214,16 +227,47 @@ export class MatTabBody implements OnInit, OnDestroy {
214
227
}
215
228
}
216
229
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
+
217
251
/** The text direction of the containing app. */
218
252
_getLayoutDirection ( ) : Direction {
219
253
return this . _dir && this . _dir . value === 'rtl' ? 'rtl' : 'ltr' ;
220
254
}
221
255
222
256
/** Whether the provided position state is considered center, regardless of origin. */
223
- _isCenterPosition ( position : MatTabBodyPositionState | string ) : boolean {
257
+ _isCenterPosition ( position : MatTabBodyPositionState | string ) : boolean {
224
258
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
+ }
227
271
}
228
272
229
273
/** Computes the position state that will be used for the tab-body animation trigger. */
0 commit comments