You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
493 lines
71 KiB
493 lines
71 KiB
/**
|
|
* @license
|
|
* Copyright Google LLC All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.io/license
|
|
*/
|
|
import { ChangeDetectorRef, ElementRef, NgZone, Optional, EventEmitter, Directive, Inject, Input, } from '@angular/core';
|
|
import { Directionality } from '@angular/cdk/bidi';
|
|
import { coerceNumberProperty } from '@angular/cdk/coercion';
|
|
import { ViewportRuler } from '@angular/cdk/scrolling';
|
|
import { FocusKeyManager } from '@angular/cdk/a11y';
|
|
import { ENTER, SPACE, hasModifierKey } from '@angular/cdk/keycodes';
|
|
import { merge, of as observableOf, Subject, timer, fromEvent } from 'rxjs';
|
|
import { takeUntil } from 'rxjs/operators';
|
|
import { Platform, normalizePassiveListenerOptions } from '@angular/cdk/platform';
|
|
import { ANIMATION_MODULE_TYPE } from '@angular/platform-browser/animations';
|
|
/** Config used to bind passive event listeners */
|
|
import * as ɵngcc0 from '@angular/core';
|
|
import * as ɵngcc1 from '@angular/cdk/scrolling';
|
|
import * as ɵngcc2 from '@angular/cdk/bidi';
|
|
import * as ɵngcc3 from '@angular/cdk/platform';
|
|
const passiveEventListenerOptions = normalizePassiveListenerOptions({ passive: true });
|
|
/**
|
|
* The distance in pixels that will be overshot when scrolling a tab label into view. This helps
|
|
* provide a small affordance to the label next to it.
|
|
*/
|
|
const EXAGGERATED_OVERSCROLL = 60;
|
|
/**
|
|
* Amount of milliseconds to wait before starting to scroll the header automatically.
|
|
* Set a little conservatively in order to handle fake events dispatched on touch devices.
|
|
*/
|
|
const HEADER_SCROLL_DELAY = 650;
|
|
/**
|
|
* Interval in milliseconds at which to scroll the header
|
|
* while the user is holding their pointer.
|
|
*/
|
|
const HEADER_SCROLL_INTERVAL = 100;
|
|
/**
|
|
* Base class for a tab header that supported pagination.
|
|
* @docs-private
|
|
*/
|
|
export class MatPaginatedTabHeader {
|
|
constructor(_elementRef, _changeDetectorRef, _viewportRuler, _dir, _ngZone, _platform, _animationMode) {
|
|
this._elementRef = _elementRef;
|
|
this._changeDetectorRef = _changeDetectorRef;
|
|
this._viewportRuler = _viewportRuler;
|
|
this._dir = _dir;
|
|
this._ngZone = _ngZone;
|
|
this._platform = _platform;
|
|
this._animationMode = _animationMode;
|
|
/** The distance in pixels that the tab labels should be translated to the left. */
|
|
this._scrollDistance = 0;
|
|
/** Whether the header should scroll to the selected index after the view has been checked. */
|
|
this._selectedIndexChanged = false;
|
|
/** Emits when the component is destroyed. */
|
|
this._destroyed = new Subject();
|
|
/** Whether the controls for pagination should be displayed */
|
|
this._showPaginationControls = false;
|
|
/** Whether the tab list can be scrolled more towards the end of the tab label list. */
|
|
this._disableScrollAfter = true;
|
|
/** Whether the tab list can be scrolled more towards the beginning of the tab label list. */
|
|
this._disableScrollBefore = true;
|
|
/** Stream that will stop the automated scrolling. */
|
|
this._stopScrolling = new Subject();
|
|
/**
|
|
* Whether pagination should be disabled. This can be used to avoid unnecessary
|
|
* layout recalculations if it's known that pagination won't be required.
|
|
*/
|
|
this.disablePagination = false;
|
|
this._selectedIndex = 0;
|
|
/** Event emitted when the option is selected. */
|
|
this.selectFocusedIndex = new EventEmitter();
|
|
/** Event emitted when a label is focused. */
|
|
this.indexFocused = new EventEmitter();
|
|
// Bind the `mouseleave` event on the outside since it doesn't change anything in the view.
|
|
_ngZone.runOutsideAngular(() => {
|
|
fromEvent(_elementRef.nativeElement, 'mouseleave')
|
|
.pipe(takeUntil(this._destroyed))
|
|
.subscribe(() => {
|
|
this._stopInterval();
|
|
});
|
|
});
|
|
}
|
|
/** The index of the active tab. */
|
|
get selectedIndex() { return this._selectedIndex; }
|
|
set selectedIndex(value) {
|
|
value = coerceNumberProperty(value);
|
|
if (this._selectedIndex != value) {
|
|
this._selectedIndexChanged = true;
|
|
this._selectedIndex = value;
|
|
if (this._keyManager) {
|
|
this._keyManager.updateActiveItem(value);
|
|
}
|
|
}
|
|
}
|
|
ngAfterViewInit() {
|
|
// We need to handle these events manually, because we want to bind passive event listeners.
|
|
fromEvent(this._previousPaginator.nativeElement, 'touchstart', passiveEventListenerOptions)
|
|
.pipe(takeUntil(this._destroyed))
|
|
.subscribe(() => {
|
|
this._handlePaginatorPress('before');
|
|
});
|
|
fromEvent(this._nextPaginator.nativeElement, 'touchstart', passiveEventListenerOptions)
|
|
.pipe(takeUntil(this._destroyed))
|
|
.subscribe(() => {
|
|
this._handlePaginatorPress('after');
|
|
});
|
|
}
|
|
ngAfterContentInit() {
|
|
const dirChange = this._dir ? this._dir.change : observableOf('ltr');
|
|
const resize = this._viewportRuler.change(150);
|
|
const realign = () => {
|
|
this.updatePagination();
|
|
this._alignInkBarToSelectedTab();
|
|
};
|
|
this._keyManager = new FocusKeyManager(this._items)
|
|
.withHorizontalOrientation(this._getLayoutDirection())
|
|
.withHomeAndEnd()
|
|
.withWrap();
|
|
this._keyManager.updateActiveItem(this._selectedIndex);
|
|
// Defer the first call in order to allow for slower browsers to lay out the elements.
|
|
// This helps in cases where the user lands directly on a page with paginated tabs.
|
|
typeof requestAnimationFrame !== 'undefined' ? requestAnimationFrame(realign) : realign();
|
|
// On dir change or window resize, realign the ink bar and update the orientation of
|
|
// the key manager if the direction has changed.
|
|
merge(dirChange, resize, this._items.changes).pipe(takeUntil(this._destroyed)).subscribe(() => {
|
|
// We need to defer this to give the browser some time to recalculate
|
|
// the element dimensions. The call has to be wrapped in `NgZone.run`,
|
|
// because the viewport change handler runs outside of Angular.
|
|
this._ngZone.run(() => Promise.resolve().then(realign));
|
|
this._keyManager.withHorizontalOrientation(this._getLayoutDirection());
|
|
});
|
|
// If there is a change in the focus key manager we need to emit the `indexFocused`
|
|
// event in order to provide a public event that notifies about focus changes. Also we realign
|
|
// the tabs container by scrolling the new focused tab into the visible section.
|
|
this._keyManager.change.pipe(takeUntil(this._destroyed)).subscribe(newFocusIndex => {
|
|
this.indexFocused.emit(newFocusIndex);
|
|
this._setTabFocus(newFocusIndex);
|
|
});
|
|
}
|
|
ngAfterContentChecked() {
|
|
// If the number of tab labels have changed, check if scrolling should be enabled
|
|
if (this._tabLabelCount != this._items.length) {
|
|
this.updatePagination();
|
|
this._tabLabelCount = this._items.length;
|
|
this._changeDetectorRef.markForCheck();
|
|
}
|
|
// If the selected index has changed, scroll to the label and check if the scrolling controls
|
|
// should be disabled.
|
|
if (this._selectedIndexChanged) {
|
|
this._scrollToLabel(this._selectedIndex);
|
|
this._checkScrollingControls();
|
|
this._alignInkBarToSelectedTab();
|
|
this._selectedIndexChanged = false;
|
|
this._changeDetectorRef.markForCheck();
|
|
}
|
|
// If the scroll distance has been changed (tab selected, focused, scroll controls activated),
|
|
// then translate the header to reflect this.
|
|
if (this._scrollDistanceChanged) {
|
|
this._updateTabScrollPosition();
|
|
this._scrollDistanceChanged = false;
|
|
this._changeDetectorRef.markForCheck();
|
|
}
|
|
}
|
|
ngOnDestroy() {
|
|
this._destroyed.next();
|
|
this._destroyed.complete();
|
|
this._stopScrolling.complete();
|
|
}
|
|
/** Handles keyboard events on the header. */
|
|
_handleKeydown(event) {
|
|
// We don't handle any key bindings with a modifier key.
|
|
if (hasModifierKey(event)) {
|
|
return;
|
|
}
|
|
switch (event.keyCode) {
|
|
case ENTER:
|
|
case SPACE:
|
|
if (this.focusIndex !== this.selectedIndex) {
|
|
this.selectFocusedIndex.emit(this.focusIndex);
|
|
this._itemSelected(event);
|
|
}
|
|
break;
|
|
default:
|
|
this._keyManager.onKeydown(event);
|
|
}
|
|
}
|
|
/**
|
|
* Callback for when the MutationObserver detects that the content has changed.
|
|
*/
|
|
_onContentChanges() {
|
|
const textContent = this._elementRef.nativeElement.textContent;
|
|
// We need to diff the text content of the header, because the MutationObserver callback
|
|
// will fire even if the text content didn't change which is inefficient and is prone
|
|
// to infinite loops if a poorly constructed expression is passed in (see #14249).
|
|
if (textContent !== this._currentTextContent) {
|
|
this._currentTextContent = textContent || '';
|
|
// The content observer runs outside the `NgZone` by default, which
|
|
// means that we need to bring the callback back in ourselves.
|
|
this._ngZone.run(() => {
|
|
this.updatePagination();
|
|
this._alignInkBarToSelectedTab();
|
|
this._changeDetectorRef.markForCheck();
|
|
});
|
|
}
|
|
}
|
|
/**
|
|
* Updates the view whether pagination should be enabled or not.
|
|
*
|
|
* WARNING: Calling this method can be very costly in terms of performance. It should be called
|
|
* as infrequently as possible from outside of the Tabs component as it causes a reflow of the
|
|
* page.
|
|
*/
|
|
updatePagination() {
|
|
this._checkPaginationEnabled();
|
|
this._checkScrollingControls();
|
|
this._updateTabScrollPosition();
|
|
}
|
|
/** Tracks which element has focus; used for keyboard navigation */
|
|
get focusIndex() {
|
|
return this._keyManager ? this._keyManager.activeItemIndex : 0;
|
|
}
|
|
/** When the focus index is set, we must manually send focus to the correct label */
|
|
set focusIndex(value) {
|
|
if (!this._isValidIndex(value) || this.focusIndex === value || !this._keyManager) {
|
|
return;
|
|
}
|
|
this._keyManager.setActiveItem(value);
|
|
}
|
|
/**
|
|
* Determines if an index is valid. If the tabs are not ready yet, we assume that the user is
|
|
* providing a valid index and return true.
|
|
*/
|
|
_isValidIndex(index) {
|
|
if (!this._items) {
|
|
return true;
|
|
}
|
|
const tab = this._items ? this._items.toArray()[index] : null;
|
|
return !!tab && !tab.disabled;
|
|
}
|
|
/**
|
|
* Sets focus on the HTML element for the label wrapper and scrolls it into the view if
|
|
* scrolling is enabled.
|
|
*/
|
|
_setTabFocus(tabIndex) {
|
|
if (this._showPaginationControls) {
|
|
this._scrollToLabel(tabIndex);
|
|
}
|
|
if (this._items && this._items.length) {
|
|
this._items.toArray()[tabIndex].focus();
|
|
// Do not let the browser manage scrolling to focus the element, this will be handled
|
|
// by using translation. In LTR, the scroll left should be 0. In RTL, the scroll width
|
|
// should be the full width minus the offset width.
|
|
const containerEl = this._tabListContainer.nativeElement;
|
|
const dir = this._getLayoutDirection();
|
|
if (dir == 'ltr') {
|
|
containerEl.scrollLeft = 0;
|
|
}
|
|
else {
|
|
containerEl.scrollLeft = containerEl.scrollWidth - containerEl.offsetWidth;
|
|
}
|
|
}
|
|
}
|
|
/** The layout direction of the containing app. */
|
|
_getLayoutDirection() {
|
|
return this._dir && this._dir.value === 'rtl' ? 'rtl' : 'ltr';
|
|
}
|
|
/** Performs the CSS transformation on the tab list that will cause the list to scroll. */
|
|
_updateTabScrollPosition() {
|
|
if (this.disablePagination) {
|
|
return;
|
|
}
|
|
const scrollDistance = this.scrollDistance;
|
|
const translateX = this._getLayoutDirection() === 'ltr' ? -scrollDistance : scrollDistance;
|
|
// Don't use `translate3d` here because we don't want to create a new layer. A new layer
|
|
// seems to cause flickering and overflow in Internet Explorer. For example, the ink bar
|
|
// and ripples will exceed the boundaries of the visible tab bar.
|
|
// See: https://github.com/angular/components/issues/10276
|
|
// We round the `transform` here, because transforms with sub-pixel precision cause some
|
|
// browsers to blur the content of the element.
|
|
this._tabList.nativeElement.style.transform = `translateX(${Math.round(translateX)}px)`;
|
|
// Setting the `transform` on IE will change the scroll offset of the parent, causing the
|
|
// position to be thrown off in some cases. We have to reset it ourselves to ensure that
|
|
// it doesn't get thrown off. Note that we scope it only to IE and Edge, because messing
|
|
// with the scroll position throws off Chrome 71+ in RTL mode (see #14689).
|
|
if (this._platform.TRIDENT || this._platform.EDGE) {
|
|
this._tabListContainer.nativeElement.scrollLeft = 0;
|
|
}
|
|
}
|
|
/** Sets the distance in pixels that the tab header should be transformed in the X-axis. */
|
|
get scrollDistance() { return this._scrollDistance; }
|
|
set scrollDistance(value) {
|
|
this._scrollTo(value);
|
|
}
|
|
/**
|
|
* Moves the tab list in the 'before' or 'after' direction (towards the beginning of the list or
|
|
* the end of the list, respectively). The distance to scroll is computed to be a third of the
|
|
* length of the tab list view window.
|
|
*
|
|
* This is an expensive call that forces a layout reflow to compute box and scroll metrics and
|
|
* should be called sparingly.
|
|
*/
|
|
_scrollHeader(direction) {
|
|
const viewLength = this._tabListContainer.nativeElement.offsetWidth;
|
|
// Move the scroll distance one-third the length of the tab list's viewport.
|
|
const scrollAmount = (direction == 'before' ? -1 : 1) * viewLength / 3;
|
|
return this._scrollTo(this._scrollDistance + scrollAmount);
|
|
}
|
|
/** Handles click events on the pagination arrows. */
|
|
_handlePaginatorClick(direction) {
|
|
this._stopInterval();
|
|
this._scrollHeader(direction);
|
|
}
|
|
/**
|
|
* Moves the tab list such that the desired tab label (marked by index) is moved into view.
|
|
*
|
|
* This is an expensive call that forces a layout reflow to compute box and scroll metrics and
|
|
* should be called sparingly.
|
|
*/
|
|
_scrollToLabel(labelIndex) {
|
|
if (this.disablePagination) {
|
|
return;
|
|
}
|
|
const selectedLabel = this._items ? this._items.toArray()[labelIndex] : null;
|
|
if (!selectedLabel) {
|
|
return;
|
|
}
|
|
// The view length is the visible width of the tab labels.
|
|
const viewLength = this._tabListContainer.nativeElement.offsetWidth;
|
|
const { offsetLeft, offsetWidth } = selectedLabel.elementRef.nativeElement;
|
|
let labelBeforePos, labelAfterPos;
|
|
if (this._getLayoutDirection() == 'ltr') {
|
|
labelBeforePos = offsetLeft;
|
|
labelAfterPos = labelBeforePos + offsetWidth;
|
|
}
|
|
else {
|
|
labelAfterPos = this._tabList.nativeElement.offsetWidth - offsetLeft;
|
|
labelBeforePos = labelAfterPos - offsetWidth;
|
|
}
|
|
const beforeVisiblePos = this.scrollDistance;
|
|
const afterVisiblePos = this.scrollDistance + viewLength;
|
|
if (labelBeforePos < beforeVisiblePos) {
|
|
// Scroll header to move label to the before direction
|
|
this.scrollDistance -= beforeVisiblePos - labelBeforePos + EXAGGERATED_OVERSCROLL;
|
|
}
|
|
else if (labelAfterPos > afterVisiblePos) {
|
|
// Scroll header to move label to the after direction
|
|
this.scrollDistance += labelAfterPos - afterVisiblePos + EXAGGERATED_OVERSCROLL;
|
|
}
|
|
}
|
|
/**
|
|
* Evaluate whether the pagination controls should be displayed. If the scroll width of the
|
|
* tab list is wider than the size of the header container, then the pagination controls should
|
|
* be shown.
|
|
*
|
|
* This is an expensive call that forces a layout reflow to compute box and scroll metrics and
|
|
* should be called sparingly.
|
|
*/
|
|
_checkPaginationEnabled() {
|
|
if (this.disablePagination) {
|
|
this._showPaginationControls = false;
|
|
}
|
|
else {
|
|
const isEnabled = this._tabList.nativeElement.scrollWidth > this._elementRef.nativeElement.offsetWidth;
|
|
if (!isEnabled) {
|
|
this.scrollDistance = 0;
|
|
}
|
|
if (isEnabled !== this._showPaginationControls) {
|
|
this._changeDetectorRef.markForCheck();
|
|
}
|
|
this._showPaginationControls = isEnabled;
|
|
}
|
|
}
|
|
/**
|
|
* Evaluate whether the before and after controls should be enabled or disabled.
|
|
* If the header is at the beginning of the list (scroll distance is equal to 0) then disable the
|
|
* before button. If the header is at the end of the list (scroll distance is equal to the
|
|
* maximum distance we can scroll), then disable the after button.
|
|
*
|
|
* This is an expensive call that forces a layout reflow to compute box and scroll metrics and
|
|
* should be called sparingly.
|
|
*/
|
|
_checkScrollingControls() {
|
|
if (this.disablePagination) {
|
|
this._disableScrollAfter = this._disableScrollBefore = true;
|
|
}
|
|
else {
|
|
// Check if the pagination arrows should be activated.
|
|
this._disableScrollBefore = this.scrollDistance == 0;
|
|
this._disableScrollAfter = this.scrollDistance == this._getMaxScrollDistance();
|
|
this._changeDetectorRef.markForCheck();
|
|
}
|
|
}
|
|
/**
|
|
* Determines what is the maximum length in pixels that can be set for the scroll distance. This
|
|
* is equal to the difference in width between the tab list container and tab header container.
|
|
*
|
|
* This is an expensive call that forces a layout reflow to compute box and scroll metrics and
|
|
* should be called sparingly.
|
|
*/
|
|
_getMaxScrollDistance() {
|
|
const lengthOfTabList = this._tabList.nativeElement.scrollWidth;
|
|
const viewLength = this._tabListContainer.nativeElement.offsetWidth;
|
|
return (lengthOfTabList - viewLength) || 0;
|
|
}
|
|
/** Tells the ink-bar to align itself to the current label wrapper */
|
|
_alignInkBarToSelectedTab() {
|
|
const selectedItem = this._items && this._items.length ?
|
|
this._items.toArray()[this.selectedIndex] : null;
|
|
const selectedLabelWrapper = selectedItem ? selectedItem.elementRef.nativeElement : null;
|
|
if (selectedLabelWrapper) {
|
|
this._inkBar.alignToElement(selectedLabelWrapper);
|
|
}
|
|
else {
|
|
this._inkBar.hide();
|
|
}
|
|
}
|
|
/** Stops the currently-running paginator interval. */
|
|
_stopInterval() {
|
|
this._stopScrolling.next();
|
|
}
|
|
/**
|
|
* Handles the user pressing down on one of the paginators.
|
|
* Starts scrolling the header after a certain amount of time.
|
|
* @param direction In which direction the paginator should be scrolled.
|
|
*/
|
|
_handlePaginatorPress(direction, mouseEvent) {
|
|
// Don't start auto scrolling for right mouse button clicks. Note that we shouldn't have to
|
|
// null check the `button`, but we do it so we don't break tests that use fake events.
|
|
if (mouseEvent && mouseEvent.button != null && mouseEvent.button !== 0) {
|
|
return;
|
|
}
|
|
// Avoid overlapping timers.
|
|
this._stopInterval();
|
|
// Start a timer after the delay and keep firing based on the interval.
|
|
timer(HEADER_SCROLL_DELAY, HEADER_SCROLL_INTERVAL)
|
|
// Keep the timer going until something tells it to stop or the component is destroyed.
|
|
.pipe(takeUntil(merge(this._stopScrolling, this._destroyed)))
|
|
.subscribe(() => {
|
|
const { maxScrollDistance, distance } = this._scrollHeader(direction);
|
|
// Stop the timer if we've reached the start or the end.
|
|
if (distance === 0 || distance >= maxScrollDistance) {
|
|
this._stopInterval();
|
|
}
|
|
});
|
|
}
|
|
/**
|
|
* Scrolls the header to a given position.
|
|
* @param position Position to which to scroll.
|
|
* @returns Information on the current scroll distance and the maximum.
|
|
*/
|
|
_scrollTo(position) {
|
|
if (this.disablePagination) {
|
|
return { maxScrollDistance: 0, distance: 0 };
|
|
}
|
|
const maxScrollDistance = this._getMaxScrollDistance();
|
|
this._scrollDistance = Math.max(0, Math.min(maxScrollDistance, position));
|
|
// Mark that the scroll distance has changed so that after the view is checked, the CSS
|
|
// transformation can move the header.
|
|
this._scrollDistanceChanged = true;
|
|
this._checkScrollingControls();
|
|
return { maxScrollDistance, distance: this._scrollDistance };
|
|
}
|
|
}
|
|
MatPaginatedTabHeader.ɵfac = function MatPaginatedTabHeader_Factory(t) { return new (t || MatPaginatedTabHeader)(ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ElementRef), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ChangeDetectorRef), ɵngcc0.ɵɵdirectiveInject(ɵngcc1.ViewportRuler), ɵngcc0.ɵɵdirectiveInject(ɵngcc2.Directionality, 8), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.NgZone), ɵngcc0.ɵɵdirectiveInject(ɵngcc3.Platform), ɵngcc0.ɵɵdirectiveInject(ANIMATION_MODULE_TYPE, 8)); };
|
|
MatPaginatedTabHeader.ɵdir = /*@__PURE__*/ ɵngcc0.ɵɵdefineDirective({ type: MatPaginatedTabHeader, inputs: { disablePagination: "disablePagination" } });
|
|
MatPaginatedTabHeader.ctorParameters = () => [
|
|
{ type: ElementRef },
|
|
{ type: ChangeDetectorRef },
|
|
{ type: ViewportRuler },
|
|
{ type: Directionality, decorators: [{ type: Optional }] },
|
|
{ type: NgZone },
|
|
{ type: Platform },
|
|
{ type: String, decorators: [{ type: Optional }, { type: Inject, args: [ANIMATION_MODULE_TYPE,] }] }
|
|
];
|
|
MatPaginatedTabHeader.propDecorators = {
|
|
disablePagination: [{ type: Input }]
|
|
};
|
|
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(MatPaginatedTabHeader, [{
|
|
type: Directive
|
|
}], function () { return [{ type: ɵngcc0.ElementRef }, { type: ɵngcc0.ChangeDetectorRef }, { type: ɵngcc1.ViewportRuler }, { type: ɵngcc2.Directionality, decorators: [{
|
|
type: Optional
|
|
}] }, { type: ɵngcc0.NgZone }, { type: ɵngcc3.Platform }, { type: String, decorators: [{
|
|
type: Optional
|
|
}, {
|
|
type: Inject,
|
|
args: [ANIMATION_MODULE_TYPE]
|
|
}] }]; }, { disablePagination: [{
|
|
type: Input
|
|
}] }); })();
|
|
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"paginated-tab-header.js","sources":["../../../../../../src/material/tabs/paginated-tab-header.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {\n  ChangeDetectorRef,\n  ElementRef,\n  NgZone,\n  Optional,\n  QueryList,\n  EventEmitter,\n  AfterContentChecked,\n  AfterContentInit,\n  AfterViewInit,\n  OnDestroy,\n  Directive,\n  Inject,\n  Input,\n} from '@angular/core';\nimport {Direction, Directionality} from '@angular/cdk/bidi';\nimport {coerceNumberProperty, NumberInput} from '@angular/cdk/coercion';\nimport {ViewportRuler} from '@angular/cdk/scrolling';\nimport {FocusKeyManager, FocusableOption} from '@angular/cdk/a11y';\nimport {ENTER, SPACE, hasModifierKey} from '@angular/cdk/keycodes';\nimport {merge, of as observableOf, Subject, timer, fromEvent} from 'rxjs';\nimport {takeUntil} from 'rxjs/operators';\nimport {Platform, normalizePassiveListenerOptions} from '@angular/cdk/platform';\nimport {ANIMATION_MODULE_TYPE} from '@angular/platform-browser/animations';\n\n\n/** Config used to bind passive event listeners */\nconst passiveEventListenerOptions =\n    normalizePassiveListenerOptions({passive: true}) as EventListenerOptions;\n\n/**\n * The directions that scrolling can go in when the header's tabs exceed the header width. 'After'\n * will scroll the header towards the end of the tabs list and 'before' will scroll towards the\n * beginning of the list.\n */\nexport type ScrollDirection = 'after' | 'before';\n\n/**\n * The distance in pixels that will be overshot when scrolling a tab label into view. This helps\n * provide a small affordance to the label next to it.\n */\nconst EXAGGERATED_OVERSCROLL = 60;\n\n/**\n * Amount of milliseconds to wait before starting to scroll the header automatically.\n * Set a little conservatively in order to handle fake events dispatched on touch devices.\n */\nconst HEADER_SCROLL_DELAY = 650;\n\n/**\n * Interval in milliseconds at which to scroll the header\n * while the user is holding their pointer.\n */\nconst HEADER_SCROLL_INTERVAL = 100;\n\n/** Item inside a paginated tab header. */\nexport type MatPaginatedTabHeaderItem = FocusableOption & {elementRef: ElementRef};\n\n/**\n * Base class for a tab header that supported pagination.\n * @docs-private\n */\n@Directive()\nexport abstract class MatPaginatedTabHeader implements AfterContentChecked, AfterContentInit,\n  AfterViewInit, OnDestroy {\n  abstract _items: QueryList<MatPaginatedTabHeaderItem>;\n  abstract _inkBar: {hide: () => void, alignToElement: (element: HTMLElement) => void};\n  abstract _tabListContainer: ElementRef<HTMLElement>;\n  abstract _tabList: ElementRef<HTMLElement>;\n  abstract _nextPaginator: ElementRef<HTMLElement>;\n  abstract _previousPaginator: ElementRef<HTMLElement>;\n\n  /** The distance in pixels that the tab labels should be translated to the left. */\n  private _scrollDistance = 0;\n\n  /** Whether the header should scroll to the selected index after the view has been checked. */\n  private _selectedIndexChanged = false;\n\n  /** Emits when the component is destroyed. */\n  protected readonly _destroyed = new Subject<void>();\n\n  /** Whether the controls for pagination should be displayed */\n  _showPaginationControls = false;\n\n  /** Whether the tab list can be scrolled more towards the end of the tab label list. */\n  _disableScrollAfter = true;\n\n  /** Whether the tab list can be scrolled more towards the beginning of the tab label list. */\n  _disableScrollBefore = true;\n\n  /**\n   * The number of tab labels that are displayed on the header. When this changes, the header\n   * should re-evaluate the scroll position.\n   */\n  private _tabLabelCount: number;\n\n  /** Whether the scroll distance has changed and should be applied after the view is checked. */\n  private _scrollDistanceChanged: boolean;\n\n  /** Used to manage focus between the tabs. */\n  private _keyManager: FocusKeyManager<MatPaginatedTabHeaderItem>;\n\n  /** Cached text content of the header. */\n  private _currentTextContent: string;\n\n  /** Stream that will stop the automated scrolling. */\n  private _stopScrolling = new Subject<void>();\n\n  /**\n   * Whether pagination should be disabled. This can be used to avoid unnecessary\n   * layout recalculations if it's known that pagination won't be required.\n   */\n  @Input()\n  disablePagination: boolean = false;\n\n  /** The index of the active tab. */\n  get selectedIndex(): number { return this._selectedIndex; }\n  set selectedIndex(value: number) {\n    value = coerceNumberProperty(value);\n\n    if (this._selectedIndex != value) {\n      this._selectedIndexChanged = true;\n      this._selectedIndex = value;\n\n      if (this._keyManager) {\n        this._keyManager.updateActiveItem(value);\n      }\n    }\n  }\n  private _selectedIndex: number = 0;\n\n  /** Event emitted when the option is selected. */\n  readonly selectFocusedIndex: EventEmitter<number> = new EventEmitter<number>();\n\n  /** Event emitted when a label is focused. */\n  readonly indexFocused: EventEmitter<number> = new EventEmitter<number>();\n\n  constructor(protected _elementRef: ElementRef<HTMLElement>,\n              protected _changeDetectorRef: ChangeDetectorRef,\n              private _viewportRuler: ViewportRuler,\n              @Optional() private _dir: Directionality,\n              private _ngZone: NgZone,\n              private _platform: Platform,\n              @Optional() @Inject(ANIMATION_MODULE_TYPE) public _animationMode?: string) {\n\n    // Bind the `mouseleave` event on the outside since it doesn't change anything in the view.\n    _ngZone.runOutsideAngular(() => {\n      fromEvent(_elementRef.nativeElement, 'mouseleave')\n        .pipe(takeUntil(this._destroyed))\n        .subscribe(() => {\n          this._stopInterval();\n        });\n    });\n  }\n\n  /** Called when the user has selected an item via the keyboard. */\n  protected abstract _itemSelected(event: KeyboardEvent): void;\n\n  ngAfterViewInit() {\n    // We need to handle these events manually, because we want to bind passive event listeners.\n    fromEvent(this._previousPaginator.nativeElement, 'touchstart', passiveEventListenerOptions)\n      .pipe(takeUntil(this._destroyed))\n      .subscribe(() => {\n        this._handlePaginatorPress('before');\n      });\n\n    fromEvent(this._nextPaginator.nativeElement, 'touchstart', passiveEventListenerOptions)\n      .pipe(takeUntil(this._destroyed))\n      .subscribe(() => {\n        this._handlePaginatorPress('after');\n      });\n  }\n\n  ngAfterContentInit() {\n    const dirChange = this._dir ? this._dir.change : observableOf('ltr');\n    const resize = this._viewportRuler.change(150);\n    const realign = () => {\n      this.updatePagination();\n      this._alignInkBarToSelectedTab();\n    };\n\n    this._keyManager = new FocusKeyManager<MatPaginatedTabHeaderItem>(this._items)\n      .withHorizontalOrientation(this._getLayoutDirection())\n      .withHomeAndEnd()\n      .withWrap();\n\n    this._keyManager.updateActiveItem(this._selectedIndex);\n\n    // Defer the first call in order to allow for slower browsers to lay out the elements.\n    // This helps in cases where the user lands directly on a page with paginated tabs.\n    typeof requestAnimationFrame !== 'undefined' ? requestAnimationFrame(realign) : realign();\n\n    // On dir change or window resize, realign the ink bar and update the orientation of\n    // the key manager if the direction has changed.\n    merge(dirChange, resize, this._items.changes).pipe(takeUntil(this._destroyed)).subscribe(() => {\n      // We need to defer this to give the browser some time to recalculate\n      // the element dimensions. The call has to be wrapped in `NgZone.run`,\n      // because the viewport change handler runs outside of Angular.\n      this._ngZone.run(() => Promise.resolve().then(realign));\n      this._keyManager.withHorizontalOrientation(this._getLayoutDirection());\n    });\n\n    // If there is a change in the focus key manager we need to emit the `indexFocused`\n    // event in order to provide a public event that notifies about focus changes. Also we realign\n    // the tabs container by scrolling the new focused tab into the visible section.\n    this._keyManager.change.pipe(takeUntil(this._destroyed)).subscribe(newFocusIndex => {\n      this.indexFocused.emit(newFocusIndex);\n      this._setTabFocus(newFocusIndex);\n    });\n  }\n\n  ngAfterContentChecked(): void {\n    // If the number of tab labels have changed, check if scrolling should be enabled\n    if (this._tabLabelCount != this._items.length) {\n      this.updatePagination();\n      this._tabLabelCount = this._items.length;\n      this._changeDetectorRef.markForCheck();\n    }\n\n    // If the selected index has changed, scroll to the label and check if the scrolling controls\n    // should be disabled.\n    if (this._selectedIndexChanged) {\n      this._scrollToLabel(this._selectedIndex);\n      this._checkScrollingControls();\n      this._alignInkBarToSelectedTab();\n      this._selectedIndexChanged = false;\n      this._changeDetectorRef.markForCheck();\n    }\n\n    // If the scroll distance has been changed (tab selected, focused, scroll controls activated),\n    // then translate the header to reflect this.\n    if (this._scrollDistanceChanged) {\n      this._updateTabScrollPosition();\n      this._scrollDistanceChanged = false;\n      this._changeDetectorRef.markForCheck();\n    }\n  }\n\n  ngOnDestroy() {\n    this._destroyed.next();\n    this._destroyed.complete();\n    this._stopScrolling.complete();\n  }\n\n  /** Handles keyboard events on the header. */\n  _handleKeydown(event: KeyboardEvent) {\n    // We don't handle any key bindings with a modifier key.\n    if (hasModifierKey(event)) {\n      return;\n    }\n\n    switch (event.keyCode) {\n      case ENTER:\n      case SPACE:\n        if (this.focusIndex !== this.selectedIndex) {\n          this.selectFocusedIndex.emit(this.focusIndex);\n          this._itemSelected(event);\n        }\n        break;\n      default:\n        this._keyManager.onKeydown(event);\n    }\n  }\n\n  /**\n   * Callback for when the MutationObserver detects that the content has changed.\n   */\n  _onContentChanges() {\n    const textContent = this._elementRef.nativeElement.textContent;\n\n    // We need to diff the text content of the header, because the MutationObserver callback\n    // will fire even if the text content didn't change which is inefficient and is prone\n    // to infinite loops if a poorly constructed expression is passed in (see #14249).\n    if (textContent !== this._currentTextContent) {\n      this._currentTextContent = textContent || '';\n\n      // The content observer runs outside the `NgZone` by default, which\n      // means that we need to bring the callback back in ourselves.\n      this._ngZone.run(() => {\n        this.updatePagination();\n        this._alignInkBarToSelectedTab();\n        this._changeDetectorRef.markForCheck();\n      });\n    }\n  }\n\n  /**\n   * Updates the view whether pagination should be enabled or not.\n   *\n   * WARNING: Calling this method can be very costly in terms of performance. It should be called\n   * as infrequently as possible from outside of the Tabs component as it causes a reflow of the\n   * page.\n   */\n  updatePagination() {\n    this._checkPaginationEnabled();\n    this._checkScrollingControls();\n    this._updateTabScrollPosition();\n  }\n\n  /** Tracks which element has focus; used for keyboard navigation */\n  get focusIndex(): number {\n    return this._keyManager ? this._keyManager.activeItemIndex! : 0;\n  }\n\n  /** When the focus index is set, we must manually send focus to the correct label */\n  set focusIndex(value: number) {\n    if (!this._isValidIndex(value) || this.focusIndex === value || !this._keyManager) {\n      return;\n    }\n\n    this._keyManager.setActiveItem(value);\n  }\n\n  /**\n   * Determines if an index is valid.  If the tabs are not ready yet, we assume that the user is\n   * providing a valid index and return true.\n   */\n  _isValidIndex(index: number): boolean {\n    if (!this._items) { return true; }\n\n    const tab = this._items ? this._items.toArray()[index] : null;\n    return !!tab && !tab.disabled;\n  }\n\n  /**\n   * Sets focus on the HTML element for the label wrapper and scrolls it into the view if\n   * scrolling is enabled.\n   */\n  _setTabFocus(tabIndex: number) {\n    if (this._showPaginationControls) {\n      this._scrollToLabel(tabIndex);\n    }\n\n    if (this._items && this._items.length) {\n      this._items.toArray()[tabIndex].focus();\n\n      // Do not let the browser manage scrolling to focus the element, this will be handled\n      // by using translation. In LTR, the scroll left should be 0. In RTL, the scroll width\n      // should be the full width minus the offset width.\n      const containerEl = this._tabListContainer.nativeElement;\n      const dir = this._getLayoutDirection();\n\n      if (dir == 'ltr') {\n        containerEl.scrollLeft = 0;\n      } else {\n        containerEl.scrollLeft = containerEl.scrollWidth - containerEl.offsetWidth;\n      }\n    }\n  }\n\n  /** The layout direction of the containing app. */\n  _getLayoutDirection(): Direction {\n    return this._dir && this._dir.value === 'rtl' ? 'rtl' : 'ltr';\n  }\n\n  /** Performs the CSS transformation on the tab list that will cause the list to scroll. */\n  _updateTabScrollPosition() {\n    if (this.disablePagination) {\n      return;\n    }\n\n    const scrollDistance = this.scrollDistance;\n    const translateX = this._getLayoutDirection() === 'ltr' ? -scrollDistance : scrollDistance;\n\n    // Don't use `translate3d` here because we don't want to create a new layer. A new layer\n    // seems to cause flickering and overflow in Internet Explorer. For example, the ink bar\n    // and ripples will exceed the boundaries of the visible tab bar.\n    // See: https://github.com/angular/components/issues/10276\n    // We round the `transform` here, because transforms with sub-pixel precision cause some\n    // browsers to blur the content of the element.\n    this._tabList.nativeElement.style.transform = `translateX(${Math.round(translateX)}px)`;\n\n    // Setting the `transform` on IE will change the scroll offset of the parent, causing the\n    // position to be thrown off in some cases. We have to reset it ourselves to ensure that\n    // it doesn't get thrown off. Note that we scope it only to IE and Edge, because messing\n    // with the scroll position throws off Chrome 71+ in RTL mode (see #14689).\n    if (this._platform.TRIDENT || this._platform.EDGE) {\n      this._tabListContainer.nativeElement.scrollLeft = 0;\n    }\n  }\n\n  /** Sets the distance in pixels that the tab header should be transformed in the X-axis. */\n  get scrollDistance(): number { return this._scrollDistance; }\n  set scrollDistance(value: number) {\n    this._scrollTo(value);\n  }\n\n  /**\n   * Moves the tab list in the 'before' or 'after' direction (towards the beginning of the list or\n   * the end of the list, respectively). The distance to scroll is computed to be a third of the\n   * length of the tab list view window.\n   *\n   * This is an expensive call that forces a layout reflow to compute box and scroll metrics and\n   * should be called sparingly.\n   */\n  _scrollHeader(direction: ScrollDirection) {\n    const viewLength = this._tabListContainer.nativeElement.offsetWidth;\n\n    // Move the scroll distance one-third the length of the tab list's viewport.\n    const scrollAmount = (direction == 'before' ? -1 : 1) * viewLength / 3;\n\n    return this._scrollTo(this._scrollDistance + scrollAmount);\n  }\n\n  /** Handles click events on the pagination arrows. */\n  _handlePaginatorClick(direction: ScrollDirection) {\n    this._stopInterval();\n    this._scrollHeader(direction);\n  }\n\n  /**\n   * Moves the tab list such that the desired tab label (marked by index) is moved into view.\n   *\n   * This is an expensive call that forces a layout reflow to compute box and scroll metrics and\n   * should be called sparingly.\n   */\n  _scrollToLabel(labelIndex: number) {\n    if (this.disablePagination) {\n      return;\n    }\n\n    const selectedLabel = this._items ? this._items.toArray()[labelIndex] : null;\n\n    if (!selectedLabel) {\n      return;\n    }\n\n    // The view length is the visible width of the tab labels.\n    const viewLength = this._tabListContainer.nativeElement.offsetWidth;\n    const {offsetLeft, offsetWidth} = selectedLabel.elementRef.nativeElement;\n\n    let labelBeforePos: number, labelAfterPos: number;\n    if (this._getLayoutDirection() == 'ltr') {\n      labelBeforePos = offsetLeft;\n      labelAfterPos = labelBeforePos + offsetWidth;\n    } else {\n      labelAfterPos = this._tabList.nativeElement.offsetWidth - offsetLeft;\n      labelBeforePos = labelAfterPos - offsetWidth;\n    }\n\n    const beforeVisiblePos = this.scrollDistance;\n    const afterVisiblePos = this.scrollDistance + viewLength;\n\n    if (labelBeforePos < beforeVisiblePos) {\n      // Scroll header to move label to the before direction\n      this.scrollDistance -= beforeVisiblePos - labelBeforePos + EXAGGERATED_OVERSCROLL;\n    } else if (labelAfterPos > afterVisiblePos) {\n      // Scroll header to move label to the after direction\n      this.scrollDistance += labelAfterPos - afterVisiblePos + EXAGGERATED_OVERSCROLL;\n    }\n  }\n\n  /**\n   * Evaluate whether the pagination controls should be displayed. If the scroll width of the\n   * tab list is wider than the size of the header container, then the pagination controls should\n   * be shown.\n   *\n   * This is an expensive call that forces a layout reflow to compute box and scroll metrics and\n   * should be called sparingly.\n   */\n  _checkPaginationEnabled() {\n    if (this.disablePagination) {\n      this._showPaginationControls = false;\n    } else {\n      const isEnabled =\n          this._tabList.nativeElement.scrollWidth > this._elementRef.nativeElement.offsetWidth;\n\n      if (!isEnabled) {\n        this.scrollDistance = 0;\n      }\n\n      if (isEnabled !== this._showPaginationControls) {\n        this._changeDetectorRef.markForCheck();\n      }\n\n      this._showPaginationControls = isEnabled;\n    }\n  }\n\n  /**\n   * Evaluate whether the before and after controls should be enabled or disabled.\n   * If the header is at the beginning of the list (scroll distance is equal to 0) then disable the\n   * before button. If the header is at the end of the list (scroll distance is equal to the\n   * maximum distance we can scroll), then disable the after button.\n   *\n   * This is an expensive call that forces a layout reflow to compute box and scroll metrics and\n   * should be called sparingly.\n   */\n  _checkScrollingControls() {\n    if (this.disablePagination) {\n      this._disableScrollAfter = this._disableScrollBefore = true;\n    } else {\n      // Check if the pagination arrows should be activated.\n      this._disableScrollBefore = this.scrollDistance == 0;\n      this._disableScrollAfter = this.scrollDistance == this._getMaxScrollDistance();\n      this._changeDetectorRef.markForCheck();\n    }\n  }\n\n  /**\n   * Determines what is the maximum length in pixels that can be set for the scroll distance. This\n   * is equal to the difference in width between the tab list container and tab header container.\n   *\n   * This is an expensive call that forces a layout reflow to compute box and scroll metrics and\n   * should be called sparingly.\n   */\n  _getMaxScrollDistance(): number {\n    const lengthOfTabList = this._tabList.nativeElement.scrollWidth;\n    const viewLength = this._tabListContainer.nativeElement.offsetWidth;\n    return (lengthOfTabList - viewLength) || 0;\n  }\n\n  /** Tells the ink-bar to align itself to the current label wrapper */\n  _alignInkBarToSelectedTab(): void {\n    const selectedItem = this._items && this._items.length ?\n        this._items.toArray()[this.selectedIndex] : null;\n    const selectedLabelWrapper = selectedItem ? selectedItem.elementRef.nativeElement : null;\n\n    if (selectedLabelWrapper) {\n      this._inkBar.alignToElement(selectedLabelWrapper);\n    } else {\n      this._inkBar.hide();\n    }\n  }\n\n  /** Stops the currently-running paginator interval.  */\n  _stopInterval() {\n    this._stopScrolling.next();\n  }\n\n  /**\n   * Handles the user pressing down on one of the paginators.\n   * Starts scrolling the header after a certain amount of time.\n   * @param direction In which direction the paginator should be scrolled.\n   */\n  _handlePaginatorPress(direction: ScrollDirection, mouseEvent?: MouseEvent) {\n    // Don't start auto scrolling for right mouse button clicks. Note that we shouldn't have to\n    // null check the `button`, but we do it so we don't break tests that use fake events.\n    if (mouseEvent && mouseEvent.button != null && mouseEvent.button !== 0) {\n      return;\n    }\n\n    // Avoid overlapping timers.\n    this._stopInterval();\n\n    // Start a timer after the delay and keep firing based on the interval.\n    timer(HEADER_SCROLL_DELAY, HEADER_SCROLL_INTERVAL)\n      // Keep the timer going until something tells it to stop or the component is destroyed.\n      .pipe(takeUntil(merge(this._stopScrolling, this._destroyed)))\n      .subscribe(() => {\n        const {maxScrollDistance, distance} = this._scrollHeader(direction);\n\n        // Stop the timer if we've reached the start or the end.\n        if (distance === 0 || distance >= maxScrollDistance) {\n          this._stopInterval();\n        }\n      });\n  }\n\n  /**\n   * Scrolls the header to a given position.\n   * @param position Position to which to scroll.\n   * @returns Information on the current scroll distance and the maximum.\n   */\n  private _scrollTo(position: number) {\n    if (this.disablePagination) {\n      return {maxScrollDistance: 0, distance: 0};\n    }\n\n    const maxScrollDistance = this._getMaxScrollDistance();\n    this._scrollDistance = Math.max(0, Math.min(maxScrollDistance, position));\n\n    // Mark that the scroll distance has changed so that after the view is checked, the CSS\n    // transformation can move the header.\n    this._scrollDistanceChanged = true;\n    this._checkScrollingControls();\n\n    return {maxScrollDistance, distance: this._scrollDistance};\n  }\n\n  static ngAcceptInputType_selectedIndex: NumberInput;\n}\n"]}
|