[6a3a178] | 1 | /**
|
---|
| 2 | * @license
|
---|
| 3 | * Copyright Google LLC All Rights Reserved.
|
---|
| 4 | *
|
---|
| 5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
| 6 | * found in the LICENSE file at https://angular.io/license
|
---|
| 7 | */
|
---|
| 8 | import { Platform } from '@angular/cdk/platform';
|
---|
| 9 | import { BasePortalOutlet, CdkPortalOutlet, } from '@angular/cdk/portal';
|
---|
| 10 | import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, NgZone, ViewChild, ViewEncapsulation, } from '@angular/core';
|
---|
| 11 | import { Subject } from 'rxjs';
|
---|
| 12 | import { take } from 'rxjs/operators';
|
---|
| 13 | import { matSnackBarAnimations } from './snack-bar-animations';
|
---|
| 14 | import { MatSnackBarConfig } from './snack-bar-config';
|
---|
| 15 | /**
|
---|
| 16 | * Internal component that wraps user-provided snack bar content.
|
---|
| 17 | * @docs-private
|
---|
| 18 | */
|
---|
| 19 | export class MatSnackBarContainer extends BasePortalOutlet {
|
---|
| 20 | constructor(_ngZone, _elementRef, _changeDetectorRef, _platform,
|
---|
| 21 | /** The snack bar configuration. */
|
---|
| 22 | snackBarConfig) {
|
---|
| 23 | super();
|
---|
| 24 | this._ngZone = _ngZone;
|
---|
| 25 | this._elementRef = _elementRef;
|
---|
| 26 | this._changeDetectorRef = _changeDetectorRef;
|
---|
| 27 | this._platform = _platform;
|
---|
| 28 | this.snackBarConfig = snackBarConfig;
|
---|
| 29 | /** The number of milliseconds to wait before announcing the snack bar's content. */
|
---|
| 30 | this._announceDelay = 150;
|
---|
| 31 | /** Whether the component has been destroyed. */
|
---|
| 32 | this._destroyed = false;
|
---|
| 33 | /** Subject for notifying that the snack bar has announced to screen readers. */
|
---|
| 34 | this._onAnnounce = new Subject();
|
---|
| 35 | /** Subject for notifying that the snack bar has exited from view. */
|
---|
| 36 | this._onExit = new Subject();
|
---|
| 37 | /** Subject for notifying that the snack bar has finished entering the view. */
|
---|
| 38 | this._onEnter = new Subject();
|
---|
| 39 | /** The state of the snack bar animations. */
|
---|
| 40 | this._animationState = 'void';
|
---|
| 41 | /**
|
---|
| 42 | * Attaches a DOM portal to the snack bar container.
|
---|
| 43 | * @deprecated To be turned into a method.
|
---|
| 44 | * @breaking-change 10.0.0
|
---|
| 45 | */
|
---|
| 46 | this.attachDomPortal = (portal) => {
|
---|
| 47 | this._assertNotAttached();
|
---|
| 48 | this._applySnackBarClasses();
|
---|
| 49 | return this._portalOutlet.attachDomPortal(portal);
|
---|
| 50 | };
|
---|
| 51 | // Use aria-live rather than a live role like 'alert' or 'status'
|
---|
| 52 | // because NVDA and JAWS have show inconsistent behavior with live roles.
|
---|
| 53 | if (snackBarConfig.politeness === 'assertive' && !snackBarConfig.announcementMessage) {
|
---|
| 54 | this._live = 'assertive';
|
---|
| 55 | }
|
---|
| 56 | else if (snackBarConfig.politeness === 'off') {
|
---|
| 57 | this._live = 'off';
|
---|
| 58 | }
|
---|
| 59 | else {
|
---|
| 60 | this._live = 'polite';
|
---|
| 61 | }
|
---|
| 62 | // Only set role for Firefox. Set role based on aria-live because setting role="alert" implies
|
---|
| 63 | // aria-live="assertive" which may cause issues if aria-live is set to "polite" above.
|
---|
| 64 | if (this._platform.FIREFOX) {
|
---|
| 65 | if (this._live === 'polite') {
|
---|
| 66 | this._role = 'status';
|
---|
| 67 | }
|
---|
| 68 | if (this._live === 'assertive') {
|
---|
| 69 | this._role = 'alert';
|
---|
| 70 | }
|
---|
| 71 | }
|
---|
| 72 | }
|
---|
| 73 | /** Attach a component portal as content to this snack bar container. */
|
---|
| 74 | attachComponentPortal(portal) {
|
---|
| 75 | this._assertNotAttached();
|
---|
| 76 | this._applySnackBarClasses();
|
---|
| 77 | return this._portalOutlet.attachComponentPortal(portal);
|
---|
| 78 | }
|
---|
| 79 | /** Attach a template portal as content to this snack bar container. */
|
---|
| 80 | attachTemplatePortal(portal) {
|
---|
| 81 | this._assertNotAttached();
|
---|
| 82 | this._applySnackBarClasses();
|
---|
| 83 | return this._portalOutlet.attachTemplatePortal(portal);
|
---|
| 84 | }
|
---|
| 85 | /** Handle end of animations, updating the state of the snackbar. */
|
---|
| 86 | onAnimationEnd(event) {
|
---|
| 87 | const { fromState, toState } = event;
|
---|
| 88 | if ((toState === 'void' && fromState !== 'void') || toState === 'hidden') {
|
---|
| 89 | this._completeExit();
|
---|
| 90 | }
|
---|
| 91 | if (toState === 'visible') {
|
---|
| 92 | // Note: we shouldn't use `this` inside the zone callback,
|
---|
| 93 | // because it can cause a memory leak.
|
---|
| 94 | const onEnter = this._onEnter;
|
---|
| 95 | this._ngZone.run(() => {
|
---|
| 96 | onEnter.next();
|
---|
| 97 | onEnter.complete();
|
---|
| 98 | });
|
---|
| 99 | }
|
---|
| 100 | }
|
---|
| 101 | /** Begin animation of snack bar entrance into view. */
|
---|
| 102 | enter() {
|
---|
| 103 | if (!this._destroyed) {
|
---|
| 104 | this._animationState = 'visible';
|
---|
| 105 | this._changeDetectorRef.detectChanges();
|
---|
| 106 | this._screenReaderAnnounce();
|
---|
| 107 | }
|
---|
| 108 | }
|
---|
| 109 | /** Begin animation of the snack bar exiting from view. */
|
---|
| 110 | exit() {
|
---|
| 111 | // Note: this one transitions to `hidden`, rather than `void`, in order to handle the case
|
---|
| 112 | // where multiple snack bars are opened in quick succession (e.g. two consecutive calls to
|
---|
| 113 | // `MatSnackBar.open`).
|
---|
| 114 | this._animationState = 'hidden';
|
---|
| 115 | // Mark this element with an 'exit' attribute to indicate that the snackbar has
|
---|
| 116 | // been dismissed and will soon be removed from the DOM. This is used by the snackbar
|
---|
| 117 | // test harness.
|
---|
| 118 | this._elementRef.nativeElement.setAttribute('mat-exit', '');
|
---|
| 119 | // If the snack bar hasn't been announced by the time it exits it wouldn't have been open
|
---|
| 120 | // long enough to visually read it either, so clear the timeout for announcing.
|
---|
| 121 | clearTimeout(this._announceTimeoutId);
|
---|
| 122 | return this._onExit;
|
---|
| 123 | }
|
---|
| 124 | /** Makes sure the exit callbacks have been invoked when the element is destroyed. */
|
---|
| 125 | ngOnDestroy() {
|
---|
| 126 | this._destroyed = true;
|
---|
| 127 | this._completeExit();
|
---|
| 128 | }
|
---|
| 129 | /**
|
---|
| 130 | * Waits for the zone to settle before removing the element. Helps prevent
|
---|
| 131 | * errors where we end up removing an element which is in the middle of an animation.
|
---|
| 132 | */
|
---|
| 133 | _completeExit() {
|
---|
| 134 | this._ngZone.onMicrotaskEmpty.pipe(take(1)).subscribe(() => {
|
---|
| 135 | this._onExit.next();
|
---|
| 136 | this._onExit.complete();
|
---|
| 137 | });
|
---|
| 138 | }
|
---|
| 139 | /** Applies the various positioning and user-configured CSS classes to the snack bar. */
|
---|
| 140 | _applySnackBarClasses() {
|
---|
| 141 | const element = this._elementRef.nativeElement;
|
---|
| 142 | const panelClasses = this.snackBarConfig.panelClass;
|
---|
| 143 | if (panelClasses) {
|
---|
| 144 | if (Array.isArray(panelClasses)) {
|
---|
| 145 | // Note that we can't use a spread here, because IE doesn't support multiple arguments.
|
---|
| 146 | panelClasses.forEach(cssClass => element.classList.add(cssClass));
|
---|
| 147 | }
|
---|
| 148 | else {
|
---|
| 149 | element.classList.add(panelClasses);
|
---|
| 150 | }
|
---|
| 151 | }
|
---|
| 152 | if (this.snackBarConfig.horizontalPosition === 'center') {
|
---|
| 153 | element.classList.add('mat-snack-bar-center');
|
---|
| 154 | }
|
---|
| 155 | if (this.snackBarConfig.verticalPosition === 'top') {
|
---|
| 156 | element.classList.add('mat-snack-bar-top');
|
---|
| 157 | }
|
---|
| 158 | }
|
---|
| 159 | /** Asserts that no content is already attached to the container. */
|
---|
| 160 | _assertNotAttached() {
|
---|
| 161 | if (this._portalOutlet.hasAttached() && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
---|
| 162 | throw Error('Attempting to attach snack bar content after content is already attached');
|
---|
| 163 | }
|
---|
| 164 | }
|
---|
| 165 | /**
|
---|
| 166 | * Starts a timeout to move the snack bar content to the live region so screen readers will
|
---|
| 167 | * announce it.
|
---|
| 168 | */
|
---|
| 169 | _screenReaderAnnounce() {
|
---|
| 170 | if (!this._announceTimeoutId) {
|
---|
| 171 | this._ngZone.runOutsideAngular(() => {
|
---|
| 172 | this._announceTimeoutId = setTimeout(() => {
|
---|
| 173 | const inertElement = this._elementRef.nativeElement.querySelector('[aria-hidden]');
|
---|
| 174 | const liveElement = this._elementRef.nativeElement.querySelector('[aria-live]');
|
---|
| 175 | if (inertElement && liveElement) {
|
---|
| 176 | // If an element in the snack bar content is focused before being moved
|
---|
| 177 | // track it and restore focus after moving to the live region.
|
---|
| 178 | let focusedElement = null;
|
---|
| 179 | if (this._platform.isBrowser &&
|
---|
| 180 | document.activeElement instanceof HTMLElement &&
|
---|
| 181 | inertElement.contains(document.activeElement)) {
|
---|
| 182 | focusedElement = document.activeElement;
|
---|
| 183 | }
|
---|
| 184 | inertElement.removeAttribute('aria-hidden');
|
---|
| 185 | liveElement.appendChild(inertElement);
|
---|
| 186 | focusedElement === null || focusedElement === void 0 ? void 0 : focusedElement.focus();
|
---|
| 187 | this._onAnnounce.next();
|
---|
| 188 | this._onAnnounce.complete();
|
---|
| 189 | }
|
---|
| 190 | }, this._announceDelay);
|
---|
| 191 | });
|
---|
| 192 | }
|
---|
| 193 | }
|
---|
| 194 | }
|
---|
| 195 | MatSnackBarContainer.decorators = [
|
---|
| 196 | { type: Component, args: [{
|
---|
| 197 | selector: 'snack-bar-container',
|
---|
| 198 | template: "<!-- Initially holds the snack bar content, will be empty after announcing to screen readers. -->\n<div aria-hidden=\"true\">\n <ng-template cdkPortalOutlet></ng-template>\n</div>\n\n<!-- Will receive the snack bar content from the non-live div, move will happen a short delay after opening -->\n<div [attr.aria-live]=\"_live\" [attr.role]=\"_role\"></div>\n",
|
---|
| 199 | // In Ivy embedded views will be change detected from their declaration place, rather than
|
---|
| 200 | // where they were stamped out. This means that we can't have the snack bar container be OnPush,
|
---|
| 201 | // because it might cause snack bars that were opened from a template not to be out of date.
|
---|
| 202 | // tslint:disable-next-line:validate-decorators
|
---|
| 203 | changeDetection: ChangeDetectionStrategy.Default,
|
---|
| 204 | encapsulation: ViewEncapsulation.None,
|
---|
| 205 | animations: [matSnackBarAnimations.snackBarState],
|
---|
| 206 | host: {
|
---|
| 207 | 'class': 'mat-snack-bar-container',
|
---|
| 208 | '[@state]': '_animationState',
|
---|
| 209 | '(@state.done)': 'onAnimationEnd($event)'
|
---|
| 210 | },
|
---|
| 211 | styles: [".mat-snack-bar-container{border-radius:4px;box-sizing:border-box;display:block;margin:24px;max-width:33vw;min-width:344px;padding:14px 16px;min-height:48px;transform-origin:center}.cdk-high-contrast-active .mat-snack-bar-container{border:solid 1px}.mat-snack-bar-handset{width:100%}.mat-snack-bar-handset .mat-snack-bar-container{margin:8px;max-width:100%;min-width:0;width:100%}\n"]
|
---|
| 212 | },] }
|
---|
| 213 | ];
|
---|
| 214 | MatSnackBarContainer.ctorParameters = () => [
|
---|
| 215 | { type: NgZone },
|
---|
| 216 | { type: ElementRef },
|
---|
| 217 | { type: ChangeDetectorRef },
|
---|
| 218 | { type: Platform },
|
---|
| 219 | { type: MatSnackBarConfig }
|
---|
| 220 | ];
|
---|
| 221 | MatSnackBarContainer.propDecorators = {
|
---|
| 222 | _portalOutlet: [{ type: ViewChild, args: [CdkPortalOutlet, { static: true },] }]
|
---|
| 223 | };
|
---|
[e29cc2e] | 224 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snack-bar-container.js","sourceRoot":"","sources":["../../../../../../src/material/snack-bar/snack-bar-container.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 {AnimationEvent} from '@angular/animations';\nimport {AriaLivePoliteness} from '@angular/cdk/a11y';\nimport {Platform} from '@angular/cdk/platform';\nimport {\n  BasePortalOutlet,\n  CdkPortalOutlet,\n  ComponentPortal,\n  TemplatePortal,\n  DomPortal,\n} from '@angular/cdk/portal';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ComponentRef,\n  ElementRef,\n  EmbeddedViewRef,\n  NgZone,\n  OnDestroy,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport {Observable, Subject} from 'rxjs';\nimport {take} from 'rxjs/operators';\nimport {matSnackBarAnimations} from './snack-bar-animations';\nimport {MatSnackBarConfig} from './snack-bar-config';\n\n/**\n * Internal interface for a snack bar container.\n * @docs-private\n */\nexport interface _SnackBarContainer {\n  snackBarConfig: MatSnackBarConfig;\n  readonly _onAnnounce: Subject<any>;\n  readonly _onExit: Subject<any>;\n  readonly _onEnter: Subject<any>;\n  enter: () => void;\n  exit: () => Observable<void>;\n  attachTemplatePortal: <C>(portal: TemplatePortal<C>) => EmbeddedViewRef<C>;\n  attachComponentPortal: <T>(portal: ComponentPortal<T>) => ComponentRef<T>;\n}\n\n/**\n * Internal component that wraps user-provided snack bar content.\n * @docs-private\n */\n@Component({\n  selector: 'snack-bar-container',\n  templateUrl: 'snack-bar-container.html',\n  styleUrls: ['snack-bar-container.css'],\n  // In Ivy embedded views will be change detected from their declaration place, rather than\n  // where they were stamped out. This means that we can't have the snack bar container be OnPush,\n  // because it might cause snack bars that were opened from a template not to be out of date.\n  // tslint:disable-next-line:validate-decorators\n  changeDetection: ChangeDetectionStrategy.Default,\n  encapsulation: ViewEncapsulation.None,\n  animations: [matSnackBarAnimations.snackBarState],\n  host: {\n    'class': 'mat-snack-bar-container',\n    '[@state]': '_animationState',\n    '(@state.done)': 'onAnimationEnd($event)'\n  },\n})\nexport class MatSnackBarContainer extends BasePortalOutlet\n    implements OnDestroy, _SnackBarContainer {\n  /** The number of milliseconds to wait before announcing the snack bar's content. */\n  private readonly _announceDelay: number = 150;\n\n  /** The timeout for announcing the snack bar's content. */\n  private _announceTimeoutId: number;\n\n  /** Whether the component has been destroyed. */\n  private _destroyed = false;\n\n  /** The portal outlet inside of this container into which the snack bar content will be loaded. */\n  @ViewChild(CdkPortalOutlet, {static: true}) _portalOutlet: CdkPortalOutlet;\n\n  /** Subject for notifying that the snack bar has announced to screen readers. */\n  readonly _onAnnounce: Subject<void> = new Subject();\n\n  /** Subject for notifying that the snack bar has exited from view. */\n  readonly _onExit: Subject<void> = new Subject();\n\n  /** Subject for notifying that the snack bar has finished entering the view. */\n  readonly _onEnter: Subject<void> = new Subject();\n\n  /** The state of the snack bar animations. */\n  _animationState = 'void';\n\n  /** aria-live value for the live region. */\n  _live: AriaLivePoliteness;\n\n  /**\n   * Role of the live region. This is only for Firefox as there is a known issue where Firefox +\n   * JAWS does not read out aria-live message.\n   */\n  _role?: 'status' | 'alert';\n\n  constructor(\n    private _ngZone: NgZone,\n    private _elementRef: ElementRef<HTMLElement>,\n    private _changeDetectorRef: ChangeDetectorRef,\n    private _platform: Platform,\n    /** The snack bar configuration. */\n    public snackBarConfig: MatSnackBarConfig) {\n\n    super();\n\n    // Use aria-live rather than a live role like 'alert' or 'status'\n    // because NVDA and JAWS have show inconsistent behavior with live roles.\n    if (snackBarConfig.politeness === 'assertive' && !snackBarConfig.announcementMessage) {\n      this._live = 'assertive';\n    } else if (snackBarConfig.politeness === 'off') {\n      this._live = 'off';\n    } else {\n      this._live = 'polite';\n    }\n\n    // Only set role for Firefox. Set role based on aria-live because setting role=\"alert\" implies\n    // aria-live=\"assertive\" which may cause issues if aria-live is set to \"polite\" above.\n    if (this._platform.FIREFOX) {\n      if (this._live === 'polite') {\n        this._role = 'status';\n      }\n      if (this._live === 'assertive') {\n        this._role = 'alert';\n      }\n    }\n  }\n\n  /** Attach a component portal as content to this snack bar container. */\n  attachComponentPortal<T>(portal: ComponentPortal<T>): ComponentRef<T> {\n    this._assertNotAttached();\n    this._applySnackBarClasses();\n    return this._portalOutlet.attachComponentPortal(portal);\n  }\n\n  /** Attach a template portal as content to this snack bar container. */\n  attachTemplatePortal<C>(portal: TemplatePortal<C>): EmbeddedViewRef<C> {\n    this._assertNotAttached();\n    this._applySnackBarClasses();\n    return this._portalOutlet.attachTemplatePortal(portal);\n  }\n\n  /**\n   * Attaches a DOM portal to the snack bar container.\n   * @deprecated To be turned into a method.\n   * @breaking-change 10.0.0\n   */\n  override attachDomPortal = (portal: DomPortal) => {\n    this._assertNotAttached();\n    this._applySnackBarClasses();\n    return this._portalOutlet.attachDomPortal(portal);\n  }\n\n  /** Handle end of animations, updating the state of the snackbar. */\n  onAnimationEnd(event: AnimationEvent) {\n    const {fromState, toState} = event;\n\n    if ((toState === 'void' && fromState !== 'void') || toState === 'hidden') {\n      this._completeExit();\n    }\n\n    if (toState === 'visible') {\n      // Note: we shouldn't use `this` inside the zone callback,\n      // because it can cause a memory leak.\n      const onEnter = this._onEnter;\n\n      this._ngZone.run(() => {\n        onEnter.next();\n        onEnter.complete();\n      });\n    }\n  }\n\n  /** Begin animation of snack bar entrance into view. */\n  enter(): void {\n    if (!this._destroyed) {\n      this._animationState = 'visible';\n      this._changeDetectorRef.detectChanges();\n      this._screenReaderAnnounce();\n    }\n  }\n\n  /** Begin animation of the snack bar exiting from view. */\n  exit(): Observable<void> {\n    // Note: this one transitions to `hidden`, rather than `void`, in order to handle the case\n    // where multiple snack bars are opened in quick succession (e.g. two consecutive calls to\n    // `MatSnackBar.open`).\n    this._animationState = 'hidden';\n\n    // Mark this element with an 'exit' attribute to indicate that the snackbar has\n    // been dismissed and will soon be removed from the DOM. This is used by the snackbar\n    // test harness.\n    this._elementRef.nativeElement.setAttribute('mat-exit', '');\n\n    // If the snack bar hasn't been announced by the time it exits it wouldn't have been open\n    // long enough to visually read it either, so clear the timeout for announcing.\n    clearTimeout(this._announceTimeoutId);\n\n    return this._onExit;\n  }\n\n  /** Makes sure the exit callbacks have been invoked when the element is destroyed. */\n  ngOnDestroy() {\n    this._destroyed = true;\n    this._completeExit();\n  }\n\n  /**\n   * Waits for the zone to settle before removing the element. Helps prevent\n   * errors where we end up removing an element which is in the middle of an animation.\n   */\n  private _completeExit() {\n    this._ngZone.onMicrotaskEmpty.pipe(take(1)).subscribe(() => {\n      this._onExit.next();\n      this._onExit.complete();\n    });\n  }\n\n  /** Applies the various positioning and user-configured CSS classes to the snack bar. */\n  private _applySnackBarClasses() {\n    const element: HTMLElement = this._elementRef.nativeElement;\n    const panelClasses = this.snackBarConfig.panelClass;\n\n    if (panelClasses) {\n      if (Array.isArray(panelClasses)) {\n        // Note that we can't use a spread here, because IE doesn't support multiple arguments.\n        panelClasses.forEach(cssClass => element.classList.add(cssClass));\n      } else {\n        element.classList.add(panelClasses);\n      }\n    }\n\n    if (this.snackBarConfig.horizontalPosition === 'center') {\n      element.classList.add('mat-snack-bar-center');\n    }\n\n    if (this.snackBarConfig.verticalPosition === 'top') {\n      element.classList.add('mat-snack-bar-top');\n    }\n  }\n\n  /** Asserts that no content is already attached to the container. */\n  private _assertNotAttached() {\n    if (this._portalOutlet.hasAttached() && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw Error('Attempting to attach snack bar content after content is already attached');\n    }\n  }\n\n  /**\n   * Starts a timeout to move the snack bar content to the live region so screen readers will\n   * announce it.\n   */\n  private _screenReaderAnnounce() {\n    if (!this._announceTimeoutId) {\n      this._ngZone.runOutsideAngular(() => {\n        this._announceTimeoutId = setTimeout(() => {\n          const inertElement = this._elementRef.nativeElement.querySelector('[aria-hidden]');\n          const liveElement = this._elementRef.nativeElement.querySelector('[aria-live]');\n\n          if (inertElement && liveElement) {\n            // If an element in the snack bar content is focused before being moved\n            // track it and restore focus after moving to the live region.\n            let focusedElement: HTMLElement | null = null;\n            if (this._platform.isBrowser &&\n                document.activeElement instanceof HTMLElement &&\n                inertElement.contains(document.activeElement)) {\n              focusedElement = document.activeElement;\n            }\n\n            inertElement.removeAttribute('aria-hidden');\n            liveElement.appendChild(inertElement);\n            focusedElement?.focus();\n\n            this._onAnnounce.next();\n            this._onAnnounce.complete();\n          }\n        }, this._announceDelay);\n      });\n    }\n  }\n}\n"]} |
---|