@@ -23,6 +23,8 @@ import {
23
23
ViewContainerRef ,
24
24
forwardRef ,
25
25
ViewChild ,
26
+ OnChanges ,
27
+ SimpleChanges ,
26
28
} from '@angular/core' ;
27
29
import { AnimationEvent } from '@angular/animations' ;
28
30
import { TemplatePortal , CdkPortalOutlet , PortalHostDirective } from '@angular/cdk/portal' ;
@@ -81,6 +83,7 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
81
83
. subscribe ( ( isCentering : boolean ) => {
82
84
if ( isCentering && ! this . hasAttached ( ) ) {
83
85
this . attach ( this . _host . _content ) ;
86
+ this . _host . _restoreScrollPosition ( ) ;
84
87
}
85
88
} ) ;
86
89
@@ -111,9 +114,13 @@ export class MatTabBodyPortal extends CdkPortalOutlet implements OnInit, OnDestr
111
114
animations : [ matTabsAnimations . translateTab ] ,
112
115
host : {
113
116
'class' : 'mat-tab-body' ,
117
+ '[class.mat-tab-body-active]' : 'active' ,
114
118
} ,
115
119
} )
116
- export class MatTabBody implements OnInit {
120
+ export class MatTabBody implements OnInit , OnChanges {
121
+ /** Element wrapping the tab's content. */
122
+ @ViewChild ( 'content' ) _contentElement : ElementRef ;
123
+
117
124
/** Event emitted when the tab begins to animate towards the center as the active tab. */
118
125
@Output ( ) readonly _onCentering : EventEmitter < number > = new EventEmitter < number > ( ) ;
119
126
@@ -132,6 +139,12 @@ export class MatTabBody implements OnInit {
132
139
/** The tab body content to display. */
133
140
@Input ( 'content' ) _content : TemplatePortal ;
134
141
142
+ /** Whether the tab is currently active. */
143
+ @Input ( ) active : boolean ;
144
+
145
+ /** Scroll position of the tab before the user switched away. */
146
+ private _lastScrollPosition = 0 ;
147
+
135
148
/** The shifted index position of the tab body, where zero represents the active center tab. */
136
149
@Input ( )
137
150
set position ( position : number ) {
@@ -180,6 +193,16 @@ export class MatTabBody implements OnInit {
180
193
}
181
194
}
182
195
196
+ ngOnChanges ( changes : SimpleChanges ) {
197
+ // Cache the scroll position before moving away from the tab. Note that this has to be done
198
+ // through change detection and as early as possible, because some browsers (namely Safari)
199
+ // will reset the scroll position when we switch from an absolute to a relative position.
200
+ if ( changes . active && changes . active . previousValue ) {
201
+ this . _lastScrollPosition = this . _elementRef . nativeElement . scrollTop ||
202
+ this . _contentElement . nativeElement . scrollTop ;
203
+ }
204
+ }
205
+
183
206
_onTranslateTabComplete ( e : AnimationEvent ) : void {
184
207
// If the transition to the center is complete, emit an event.
185
208
if ( this . _isCenterPosition ( e . toState ) && this . _isCenterPosition ( this . _position ) ) {
@@ -197,9 +220,19 @@ export class MatTabBody implements OnInit {
197
220
}
198
221
199
222
/** Whether the provided position state is considered center, regardless of origin. */
200
- _isCenterPosition ( position : MatTabBodyPositionState | string ) : boolean {
223
+ _isCenterPosition ( position : MatTabBodyPositionState | string ) : boolean {
201
224
return position == 'center' ||
202
- position == 'left-origin-center' ||
203
- position == 'right-origin-center' ;
225
+ position == 'left-origin-center' ||
226
+ position == 'right-origin-center' ;
227
+ }
228
+
229
+ _restoreScrollPosition ( ) {
230
+ if ( this . _lastScrollPosition ) {
231
+ // Depending on the browser, the scrollable element can end up being
232
+ // either the host element or the element with all the content.
233
+ this . _contentElement . nativeElement . scrollTop =
234
+ this . _elementRef . nativeElement . scrollTop =
235
+ this . _lastScrollPosition ;
236
+ }
204
237
}
205
238
}
0 commit comments