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