@villedemontreal/angular-ui 13.2.1 → 13.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (37) hide show
  1. package/esm2020/lib/bao.module.mjs +12 -7
  2. package/esm2020/lib/core/breakpoints.mjs +13 -0
  3. package/esm2020/lib/core/index.mjs +8 -0
  4. package/esm2020/lib/dropdown-menu/dropdown-menu.component.mjs +10 -3
  5. package/esm2020/lib/modal/modal-container.mjs +2 -2
  6. package/esm2020/lib/snack-bar/index.mjs +12 -0
  7. package/esm2020/lib/snack-bar/module.mjs +54 -0
  8. package/esm2020/lib/snack-bar/simple-snack-bar.component.mjs +92 -0
  9. package/esm2020/lib/snack-bar/snack-bar-animations.mjs +28 -0
  10. package/esm2020/lib/snack-bar/snack-bar-config.mjs +37 -0
  11. package/esm2020/lib/snack-bar/snack-bar-container.mjs +272 -0
  12. package/esm2020/lib/snack-bar/snack-bar-ref.mjs +75 -0
  13. package/esm2020/lib/snack-bar/snack-bar.mjs +251 -0
  14. package/esm2020/lib/system-header/index.mjs +8 -0
  15. package/esm2020/lib/system-header/module.mjs +33 -0
  16. package/esm2020/lib/system-header/system-header.component.mjs +128 -0
  17. package/esm2020/public-api.mjs +3 -1
  18. package/fesm2015/villedemontreal-angular-ui.mjs +981 -15
  19. package/fesm2015/villedemontreal-angular-ui.mjs.map +1 -1
  20. package/fesm2020/villedemontreal-angular-ui.mjs +977 -15
  21. package/fesm2020/villedemontreal-angular-ui.mjs.map +1 -1
  22. package/lib/bao.module.d.ts +3 -1
  23. package/lib/core/breakpoints.d.ts +7 -0
  24. package/lib/core/index.d.ts +2 -0
  25. package/lib/snack-bar/index.d.ts +6 -0
  26. package/lib/snack-bar/module.d.ts +13 -0
  27. package/lib/snack-bar/simple-snack-bar.component.d.ts +51 -0
  28. package/lib/snack-bar/snack-bar-animations.d.ts +8 -0
  29. package/lib/snack-bar/snack-bar-config.d.ts +51 -0
  30. package/lib/snack-bar/snack-bar-container.d.ts +111 -0
  31. package/lib/snack-bar/snack-bar-ref.d.ts +51 -0
  32. package/lib/snack-bar/snack-bar.d.ts +89 -0
  33. package/lib/system-header/index.d.ts +2 -0
  34. package/lib/system-header/module.d.ts +9 -0
  35. package/lib/system-header/system-header.component.d.ts +41 -0
  36. package/package.json +1 -1
  37. package/public-api.d.ts +2 -0
@@ -0,0 +1,92 @@
1
+ /*
2
+ * Copyright (c) 2023 Ville de Montreal. All rights reserved.
3
+ * Licensed under the MIT license.
4
+ * See LICENSE file in the project root for full license information.
5
+ */
6
+ import { ChangeDetectionStrategy, Component, Inject, ViewEncapsulation } from '@angular/core';
7
+ import { ICONS_DCT } from '../icon/icons-dictionary';
8
+ import { BAO_SNACK_BAR_DATA } from './snack-bar-config';
9
+ import * as i0 from "@angular/core";
10
+ import * as i1 from "./snack-bar-ref";
11
+ import * as i2 from "../icon/icon.component";
12
+ import * as i3 from "../button/button.component";
13
+ import * as i4 from "@angular/common";
14
+ const toastTypeToAttributes = {
15
+ info: {
16
+ toastClass: 'bao-snackbar-info',
17
+ icon: 'icon-info',
18
+ iconTitle: 'Information',
19
+ politeness: 'assertive'
20
+ },
21
+ success: {
22
+ toastClass: 'bao-snackbar-success',
23
+ icon: 'icon-check-circle',
24
+ iconTitle: 'Succès',
25
+ politeness: 'polite'
26
+ },
27
+ danger: {
28
+ toastClass: 'bao-snackbar-danger',
29
+ icon: 'icon-error',
30
+ iconTitle: 'Erreur',
31
+ politeness: 'assertive'
32
+ }
33
+ };
34
+ /**
35
+ * A component used to open as the default snack bar, matching material spec.
36
+ * This should only be used internally by the snack bar service.
37
+ */
38
+ export class BaoSimpleSnackBarComponent {
39
+ constructor(snackBarRef, data) {
40
+ this.snackBarRef = snackBarRef;
41
+ this.showCloseTitle = 'Fermer le message';
42
+ this.data = data;
43
+ }
44
+ /** Returns the politeness */
45
+ get politeness() {
46
+ return (toastTypeToAttributes[this.data.toastType]?.politeness ||
47
+ toastTypeToAttributes['info'].politeness);
48
+ }
49
+ /** Returns the toast class */
50
+ get toastClass() {
51
+ return (toastTypeToAttributes[this.data.toastType]?.toastClass ||
52
+ toastTypeToAttributes['info'].toastClass);
53
+ }
54
+ /** Returns the toast icon */
55
+ get toastIcon() {
56
+ return (toastTypeToAttributes[this.data.toastType]?.icon ||
57
+ toastTypeToAttributes['info'].icon);
58
+ }
59
+ /** Returns the toast icon title */
60
+ get toastIconTitle() {
61
+ return (toastTypeToAttributes[this.data.toastType]?.iconTitle ||
62
+ toastTypeToAttributes['info'].iconTitle);
63
+ }
64
+ /** If the action button should be shown. */
65
+ get hasAction() {
66
+ return !!this.data.actionLabelOrIcon;
67
+ }
68
+ /** If the action is an icon */
69
+ get isActionIcon() {
70
+ return !!ICONS_DCT[this.data.actionLabelOrIcon];
71
+ }
72
+ /** Performs the action on the snack bar. */
73
+ action() {
74
+ this.snackBarRef.dismissWithAction();
75
+ }
76
+ /** Closes the snack bar. */
77
+ close() {
78
+ this.snackBarRef.dismiss();
79
+ }
80
+ }
81
+ BaoSimpleSnackBarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoSimpleSnackBarComponent, deps: [{ token: i1.BaoSnackBarRef }, { token: BAO_SNACK_BAR_DATA }], target: i0.ɵɵFactoryTarget.Component });
82
+ BaoSimpleSnackBarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: BaoSimpleSnackBarComponent, selector: "bao-simple-snack-bar", host: { classAttribute: "bao-simple-snackbar" }, ngImport: i0, template: "<div\n class=\"bao-snackbar show\"\n [ngClass]=\"toastClass\"\n [attr.aria-live]=\"politeness\"\n aria-atomic=\"true\"\n>\n <div class=\"bao-snackbar-icon\">\n <bao-icon [svgIcon]=\"toastIcon\" [title]=\"toastIconTitle\"></bao-icon>\n </div>\n <div class=\"bao-snackbar-body\">\n {{ data.message }}\n </div>\n <div class=\"bao-snackbar-action\">\n <div *ngIf=\"hasAction\">\n <button\n bao-button\n role=\"button\"\n type=\"utility\"\n level=\"tertiary\"\n [reversed]=\"true\"\n [title]=\"data.actionLabelOrIcon\"\n (click)=\"action()\"\n >\n <bao-icon\n *ngIf=\"isActionIcon; else isActionText\"\n [svgIcon]=\"data.actionLabelOrIcon\"\n [title]=\"data.actionLabelOrIcon\"\n ></bao-icon>\n <ng-template #isActionText\n ><span>{{ data.actionLabelOrIcon }}</span></ng-template\n >\n </button>\n </div>\n <div *ngIf=\"data.showClose\">\n <button\n bao-button\n role=\"button\"\n type=\"utility\"\n level=\"tertiary\"\n [reversed]=\"true\"\n title=\"{{ showCloseTitle }}\"\n (click)=\"close()\"\n >\n <bao-icon svgIcon=\"icon-x\" title=\"{{ showCloseTitle }}\"></bao-icon>\n </button>\n </div>\n </div>\n</div>\n", styles: [".bao-container{padding-right:16px;padding-left:16px;margin-right:auto;margin-left:auto;width:100%}@media (min-width: 576px){.bao-container{max-width:576px}}@media (min-width: 768px){.bao-container{max-width:768px}}@media (min-width: 992px){.bao-container{max-width:992px}}@media (min-width: 1200px){.bao-container{max-width:1200px}}.bao-row{display:flex;flex-wrap:wrap;margin-right:-16px;margin-left:-16px}.bao-col-12,.bao-col-lg-7{position:relative;width:100%;padding-right:1rem;padding-left:1rem}@media (min-width: 992px){.bao-col-lg-7{flex:0 0 58.33333%;max-width:58.33333%}}.bao-snackbar{overflow:hidden;font-size:.875rem;line-height:1.25rem;color:#fff;background-color:#004b7b;background-clip:padding-box;box-shadow:0 2px 8px #0000001a;-webkit-backdrop-filter:blur(10px);backdrop-filter:blur(10px);opacity:0;border-radius:.25rem;width:100%}.bao-snackbar:not(:last-child){margin-bottom:1rem}.bao-snackbar.showing{opacity:1}.bao-snackbar.show{display:inline-flex;opacity:1}.bao-snackbar.hide{display:none}@media (min-width: 768px){.bao-snackbar{margin:1rem;width:auto}}.bao-snackbar-icon{display:inline-flex;align-items:center;flex-grow:0;margin:1rem 0 1rem 1rem}.bao-snackbar-body{display:flex;align-items:center;padding:1rem;margin:0}.bao-snackbar-action{display:flex;align-items:center;margin-left:auto}.bao-snackbar-action:last-child{margin-right:.5rem}.bao-snackbar-info{color:#fff;background-color:#004b7b}.bao-snackbar-success{color:#fff;background-color:#025d29}.bao-snackbar-danger{color:#fff;background-color:#851a00}\n"], components: [{ type: i2.BaoIconComponent, selector: "bao-icon", inputs: ["color", "size", "svgIcon", "title"], exportAs: ["baoIcon"] }, { type: i3.BaoButtonComponent, selector: "button[bao-button]", inputs: ["displayType", "level", "size", "loading", "reversed", "loadingSpinnerAriaLabel", "fullWidth"] }], directives: [{ type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
83
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoSimpleSnackBarComponent, decorators: [{
84
+ type: Component,
85
+ args: [{ selector: 'bao-simple-snack-bar', encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, host: {
86
+ class: 'bao-simple-snackbar'
87
+ }, template: "<div\n class=\"bao-snackbar show\"\n [ngClass]=\"toastClass\"\n [attr.aria-live]=\"politeness\"\n aria-atomic=\"true\"\n>\n <div class=\"bao-snackbar-icon\">\n <bao-icon [svgIcon]=\"toastIcon\" [title]=\"toastIconTitle\"></bao-icon>\n </div>\n <div class=\"bao-snackbar-body\">\n {{ data.message }}\n </div>\n <div class=\"bao-snackbar-action\">\n <div *ngIf=\"hasAction\">\n <button\n bao-button\n role=\"button\"\n type=\"utility\"\n level=\"tertiary\"\n [reversed]=\"true\"\n [title]=\"data.actionLabelOrIcon\"\n (click)=\"action()\"\n >\n <bao-icon\n *ngIf=\"isActionIcon; else isActionText\"\n [svgIcon]=\"data.actionLabelOrIcon\"\n [title]=\"data.actionLabelOrIcon\"\n ></bao-icon>\n <ng-template #isActionText\n ><span>{{ data.actionLabelOrIcon }}</span></ng-template\n >\n </button>\n </div>\n <div *ngIf=\"data.showClose\">\n <button\n bao-button\n role=\"button\"\n type=\"utility\"\n level=\"tertiary\"\n [reversed]=\"true\"\n title=\"{{ showCloseTitle }}\"\n (click)=\"close()\"\n >\n <bao-icon svgIcon=\"icon-x\" title=\"{{ showCloseTitle }}\"></bao-icon>\n </button>\n </div>\n </div>\n</div>\n", styles: [".bao-container{padding-right:16px;padding-left:16px;margin-right:auto;margin-left:auto;width:100%}@media (min-width: 576px){.bao-container{max-width:576px}}@media (min-width: 768px){.bao-container{max-width:768px}}@media (min-width: 992px){.bao-container{max-width:992px}}@media (min-width: 1200px){.bao-container{max-width:1200px}}.bao-row{display:flex;flex-wrap:wrap;margin-right:-16px;margin-left:-16px}.bao-col-12,.bao-col-lg-7{position:relative;width:100%;padding-right:1rem;padding-left:1rem}@media (min-width: 992px){.bao-col-lg-7{flex:0 0 58.33333%;max-width:58.33333%}}.bao-snackbar{overflow:hidden;font-size:.875rem;line-height:1.25rem;color:#fff;background-color:#004b7b;background-clip:padding-box;box-shadow:0 2px 8px #0000001a;-webkit-backdrop-filter:blur(10px);backdrop-filter:blur(10px);opacity:0;border-radius:.25rem;width:100%}.bao-snackbar:not(:last-child){margin-bottom:1rem}.bao-snackbar.showing{opacity:1}.bao-snackbar.show{display:inline-flex;opacity:1}.bao-snackbar.hide{display:none}@media (min-width: 768px){.bao-snackbar{margin:1rem;width:auto}}.bao-snackbar-icon{display:inline-flex;align-items:center;flex-grow:0;margin:1rem 0 1rem 1rem}.bao-snackbar-body{display:flex;align-items:center;padding:1rem;margin:0}.bao-snackbar-action{display:flex;align-items:center;margin-left:auto}.bao-snackbar-action:last-child{margin-right:.5rem}.bao-snackbar-info{color:#fff;background-color:#004b7b}.bao-snackbar-success{color:#fff;background-color:#025d29}.bao-snackbar-danger{color:#fff;background-color:#851a00}\n"] }]
88
+ }], ctorParameters: function () { return [{ type: i1.BaoSnackBarRef }, { type: undefined, decorators: [{
89
+ type: Inject,
90
+ args: [BAO_SNACK_BAR_DATA]
91
+ }] }]; } });
92
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"simple-snack-bar.component.js","sourceRoot":"","sources":["../../../../../projects/angular-ui/src/lib/snack-bar/simple-snack-bar.component.ts","../../../../../projects/angular-ui/src/lib/snack-bar/simple-snack-bar.component.html"],"names":[],"mappings":"AAAA;;;;GAIG;AACH,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,MAAM,EACN,iBAAiB,EAClB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,SAAS,EAAE,MAAM,0BAA0B,CAAC;AACrD,OAAO,EAAwB,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;;;;;;AAG9E,MAAM,qBAAqB,GAAG;IAC5B,IAAI,EAAE;QACJ,UAAU,EAAE,mBAAmB;QAC/B,IAAI,EAAE,WAAW;QACjB,SAAS,EAAE,aAAa;QACxB,UAAU,EAAE,WAAW;KACxB;IACD,OAAO,EAAE;QACP,UAAU,EAAE,sBAAsB;QAClC,IAAI,EAAE,mBAAmB;QACzB,SAAS,EAAE,QAAQ;QACnB,UAAU,EAAE,QAAQ;KACrB;IACD,MAAM,EAAE;QACN,UAAU,EAAE,qBAAqB;QACjC,IAAI,EAAE,YAAY;QAClB,SAAS,EAAE,QAAQ;QACnB,UAAU,EAAE,WAAW;KACxB;CACF,CAAC;AAiBF;;;GAGG;AAWH,MAAM,OAAO,0BAA0B;IAWrC,YACS,WAAuD,EAClC,IAAS;QAD9B,gBAAW,GAAX,WAAW,CAA4C;QAXzD,mBAAc,GAAG,mBAAmB,CAAC;QAc1C,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;IACnB,CAAC;IAED,6BAA6B;IAC7B,IAAI,UAAU;QACZ,OAAO,CACL,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU;YACtD,qBAAqB,CAAC,MAAM,CAAC,CAAC,UAAU,CACzC,CAAC;IACJ,CAAC;IAED,8BAA8B;IAC9B,IAAI,UAAU;QACZ,OAAO,CACL,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU;YACtD,qBAAqB,CAAC,MAAM,CAAC,CAAC,UAAU,CACzC,CAAC;IACJ,CAAC;IAED,6BAA6B;IAC7B,IAAI,SAAS;QACX,OAAO,CACL,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI;YAChD,qBAAqB,CAAC,MAAM,CAAC,CAAC,IAAI,CACnC,CAAC;IACJ,CAAC;IAED,mCAAmC;IACnC,IAAI,cAAc;QAChB,OAAO,CACL,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,SAAS;YACrD,qBAAqB,CAAC,MAAM,CAAC,CAAC,SAAS,CACxC,CAAC;IACJ,CAAC;IAED,4CAA4C;IAC5C,IAAI,SAAS;QACX,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC;IACvC,CAAC;IAED,+BAA+B;IAC/B,IAAI,YAAY;QACd,OAAO,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;IAClD,CAAC;IAED,4CAA4C;IACrC,MAAM;QACX,IAAI,CAAC,WAAW,CAAC,iBAAiB,EAAE,CAAC;IACvC,CAAC;IAED,4BAA4B;IACrB,KAAK;QACV,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;IAC7B,CAAC;;wHApEU,0BAA0B,gDAa3B,kBAAkB;4GAbjB,0BAA0B,6GCjEvC,yzCAgDA;4FDiBa,0BAA0B;kBAVtC,SAAS;+BACE,sBAAsB,iBAGjB,iBAAiB,CAAC,IAAI,mBACpB,uBAAuB,CAAC,MAAM,QACzC;wBACJ,KAAK,EAAE,qBAAqB;qBAC7B;;0BAeE,MAAM;2BAAC,kBAAkB","sourcesContent":["/*\n * Copyright (c) 2023 Ville de Montreal. All rights reserved.\n * Licensed under the MIT license.\n * See LICENSE file in the project root for full license information.\n */\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  Inject,\n  ViewEncapsulation\n} from '@angular/core';\nimport { ICONS_DCT } from '../icon/icons-dictionary';\nimport { BaoSnackBarToastType, BAO_SNACK_BAR_DATA } from './snack-bar-config';\nimport { BaoSnackBarRef } from './snack-bar-ref';\n\nconst toastTypeToAttributes = {\n  info: {\n    toastClass: 'bao-snackbar-info',\n    icon: 'icon-info',\n    iconTitle: 'Information',\n    politeness: 'assertive'\n  },\n  success: {\n    toastClass: 'bao-snackbar-success',\n    icon: 'icon-check-circle',\n    iconTitle: 'Succès',\n    politeness: 'polite'\n  },\n  danger: {\n    toastClass: 'bao-snackbar-danger',\n    icon: 'icon-error',\n    iconTitle: 'Erreur',\n    politeness: 'assertive'\n  }\n};\n\n/**\n * Interface for a simple snack bar component that has a message and a single action.\n */\nexport interface ITextOnlySnackBar {\n  data: {\n    message: string;\n    toastType: BaoSnackBarToastType;\n    actionLabelOrIcon: string;\n    showClose: boolean;\n  };\n  snackBarRef: BaoSnackBarRef<ITextOnlySnackBar>;\n  action: () => void;\n  hasAction: boolean;\n}\n\n/**\n * A component used to open as the default snack bar, matching material spec.\n * This should only be used internally by the snack bar service.\n */\n@Component({\n  selector: 'bao-simple-snack-bar',\n  templateUrl: 'simple-snack-bar.component.html',\n  styleUrls: ['simple-snack-bar.component.scss'],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: {\n    class: 'bao-simple-snackbar'\n  }\n})\nexport class BaoSimpleSnackBarComponent implements ITextOnlySnackBar {\n  public showCloseTitle = 'Fermer le message';\n\n  /** Data that was injected into the snack bar. */\n  public data: {\n    message: string;\n    toastType: BaoSnackBarToastType;\n    actionLabelOrIcon: string;\n    showClose: boolean;\n  };\n\n  constructor(\n    public snackBarRef: BaoSnackBarRef<BaoSimpleSnackBarComponent>,\n    @Inject(BAO_SNACK_BAR_DATA) data: any\n  ) {\n    this.data = data;\n  }\n\n  /** Returns the politeness */\n  get politeness(): string {\n    return (\n      toastTypeToAttributes[this.data.toastType]?.politeness ||\n      toastTypeToAttributes['info'].politeness\n    );\n  }\n\n  /** Returns the toast class */\n  get toastClass(): string {\n    return (\n      toastTypeToAttributes[this.data.toastType]?.toastClass ||\n      toastTypeToAttributes['info'].toastClass\n    );\n  }\n\n  /** Returns the toast icon */\n  get toastIcon(): string {\n    return (\n      toastTypeToAttributes[this.data.toastType]?.icon ||\n      toastTypeToAttributes['info'].icon\n    );\n  }\n\n  /** Returns the toast icon title */\n  get toastIconTitle(): string {\n    return (\n      toastTypeToAttributes[this.data.toastType]?.iconTitle ||\n      toastTypeToAttributes['info'].iconTitle\n    );\n  }\n\n  /** If the action button should be shown. */\n  get hasAction(): boolean {\n    return !!this.data.actionLabelOrIcon;\n  }\n\n  /** If the action is an icon */\n  get isActionIcon(): boolean {\n    return !!ICONS_DCT[this.data.actionLabelOrIcon];\n  }\n\n  /** Performs the action on the snack bar. */\n  public action(): void {\n    this.snackBarRef.dismissWithAction();\n  }\n\n  /** Closes the snack bar. */\n  public close(): void {\n    this.snackBarRef.dismiss();\n  }\n}\n","<div\n  class=\"bao-snackbar show\"\n  [ngClass]=\"toastClass\"\n  [attr.aria-live]=\"politeness\"\n  aria-atomic=\"true\"\n>\n  <div class=\"bao-snackbar-icon\">\n    <bao-icon [svgIcon]=\"toastIcon\" [title]=\"toastIconTitle\"></bao-icon>\n  </div>\n  <div class=\"bao-snackbar-body\">\n    {{ data.message }}\n  </div>\n  <div class=\"bao-snackbar-action\">\n    <div *ngIf=\"hasAction\">\n      <button\n        bao-button\n        role=\"button\"\n        type=\"utility\"\n        level=\"tertiary\"\n        [reversed]=\"true\"\n        [title]=\"data.actionLabelOrIcon\"\n        (click)=\"action()\"\n      >\n        <bao-icon\n          *ngIf=\"isActionIcon; else isActionText\"\n          [svgIcon]=\"data.actionLabelOrIcon\"\n          [title]=\"data.actionLabelOrIcon\"\n        ></bao-icon>\n        <ng-template #isActionText\n          ><span>{{ data.actionLabelOrIcon }}</span></ng-template\n        >\n      </button>\n    </div>\n    <div *ngIf=\"data.showClose\">\n      <button\n        bao-button\n        role=\"button\"\n        type=\"utility\"\n        level=\"tertiary\"\n        [reversed]=\"true\"\n        title=\"{{ showCloseTitle }}\"\n        (click)=\"close()\"\n      >\n        <bao-icon svgIcon=\"icon-x\" title=\"{{ showCloseTitle }}\"></bao-icon>\n      </button>\n    </div>\n  </div>\n</div>\n"]}
@@ -0,0 +1,28 @@
1
+ /*
2
+ * Copyright (c) 2023 Ville de Montreal. All rights reserved.
3
+ * Licensed under the MIT license.
4
+ * See LICENSE file in the project root for full license information.
5
+ */
6
+ import { animate, state, style, transition, trigger } from '@angular/animations';
7
+ /**
8
+ * Animations used by the Material snack bar.
9
+ * @docs-private
10
+ */
11
+ export const matSnackBarAnimations = {
12
+ /** Animation that shows and hides a snack bar. */
13
+ snackBarState: trigger('state', [
14
+ state('void, hidden', style({
15
+ transform: 'scale(0.8)',
16
+ opacity: 0
17
+ })),
18
+ state('visible', style({
19
+ transform: 'scale(1)',
20
+ opacity: 1
21
+ })),
22
+ transition('* => visible', animate('150ms cubic-bezier(0, 0, 0.2, 1)')),
23
+ transition('* => void, * => hidden', animate('75ms cubic-bezier(0.4, 0.0, 1, 1)', style({
24
+ opacity: 0
25
+ })))
26
+ ])
27
+ };
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic25hY2stYmFyLWFuaW1hdGlvbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hbmd1bGFyLXVpL3NyYy9saWIvc25hY2stYmFyL3NuYWNrLWJhci1hbmltYXRpb25zLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFDSCxPQUFPLEVBQ0wsT0FBTyxFQUVQLEtBQUssRUFDTCxLQUFLLEVBQ0wsVUFBVSxFQUNWLE9BQU8sRUFDUixNQUFNLHFCQUFxQixDQUFDO0FBRTdCOzs7R0FHRztBQUNILE1BQU0sQ0FBQyxNQUFNLHFCQUFxQixHQUU5QjtJQUNGLGtEQUFrRDtJQUNsRCxhQUFhLEVBQUUsT0FBTyxDQUFDLE9BQU8sRUFBRTtRQUM5QixLQUFLLENBQ0gsY0FBYyxFQUNkLEtBQUssQ0FBQztZQUNKLFNBQVMsRUFBRSxZQUFZO1lBQ3ZCLE9BQU8sRUFBRSxDQUFDO1NBQ1gsQ0FBQyxDQUNIO1FBQ0QsS0FBSyxDQUNILFNBQVMsRUFDVCxLQUFLLENBQUM7WUFDSixTQUFTLEVBQUUsVUFBVTtZQUNyQixPQUFPLEVBQUUsQ0FBQztTQUNYLENBQUMsQ0FDSDtRQUNELFVBQVUsQ0FBQyxjQUFjLEVBQUUsT0FBTyxDQUFDLGtDQUFrQyxDQUFDLENBQUM7UUFDdkUsVUFBVSxDQUNSLHdCQUF3QixFQUN4QixPQUFPLENBQ0wsbUNBQW1DLEVBQ25DLEtBQUssQ0FBQztZQUNKLE9BQU8sRUFBRSxDQUFDO1NBQ1gsQ0FBQyxDQUNILENBQ0Y7S0FDRixDQUFDO0NBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjMgVmlsbGUgZGUgTW9udHJlYWwuIEFsbCByaWdodHMgcmVzZXJ2ZWQuXG4gKiBMaWNlbnNlZCB1bmRlciB0aGUgTUlUIGxpY2Vuc2UuXG4gKiBTZWUgTElDRU5TRSBmaWxlIGluIHRoZSBwcm9qZWN0IHJvb3QgZm9yIGZ1bGwgbGljZW5zZSBpbmZvcm1hdGlvbi5cbiAqL1xuaW1wb3J0IHtcbiAgYW5pbWF0ZSxcbiAgQW5pbWF0aW9uVHJpZ2dlck1ldGFkYXRhLFxuICBzdGF0ZSxcbiAgc3R5bGUsXG4gIHRyYW5zaXRpb24sXG4gIHRyaWdnZXJcbn0gZnJvbSAnQGFuZ3VsYXIvYW5pbWF0aW9ucyc7XG5cbi8qKlxuICogQW5pbWF0aW9ucyB1c2VkIGJ5IHRoZSBNYXRlcmlhbCBzbmFjayBiYXIuXG4gKiBAZG9jcy1wcml2YXRlXG4gKi9cbmV4cG9ydCBjb25zdCBtYXRTbmFja0JhckFuaW1hdGlvbnM6IHtcbiAgcmVhZG9ubHkgc25hY2tCYXJTdGF0ZTogQW5pbWF0aW9uVHJpZ2dlck1ldGFkYXRhO1xufSA9IHtcbiAgLyoqIEFuaW1hdGlvbiB0aGF0IHNob3dzIGFuZCBoaWRlcyBhIHNuYWNrIGJhci4gKi9cbiAgc25hY2tCYXJTdGF0ZTogdHJpZ2dlcignc3RhdGUnLCBbXG4gICAgc3RhdGUoXG4gICAgICAndm9pZCwgaGlkZGVuJyxcbiAgICAgIHN0eWxlKHtcbiAgICAgICAgdHJhbnNmb3JtOiAnc2NhbGUoMC44KScsXG4gICAgICAgIG9wYWNpdHk6IDBcbiAgICAgIH0pXG4gICAgKSxcbiAgICBzdGF0ZShcbiAgICAgICd2aXNpYmxlJyxcbiAgICAgIHN0eWxlKHtcbiAgICAgICAgdHJhbnNmb3JtOiAnc2NhbGUoMSknLFxuICAgICAgICBvcGFjaXR5OiAxXG4gICAgICB9KVxuICAgICksXG4gICAgdHJhbnNpdGlvbignKiA9PiB2aXNpYmxlJywgYW5pbWF0ZSgnMTUwbXMgY3ViaWMtYmV6aWVyKDAsIDAsIDAuMiwgMSknKSksXG4gICAgdHJhbnNpdGlvbihcbiAgICAgICcqID0+IHZvaWQsICogPT4gaGlkZGVuJyxcbiAgICAgIGFuaW1hdGUoXG4gICAgICAgICc3NW1zIGN1YmljLWJlemllcigwLjQsIDAuMCwgMSwgMSknLFxuICAgICAgICBzdHlsZSh7XG4gICAgICAgICAgb3BhY2l0eTogMFxuICAgICAgICB9KVxuICAgICAgKVxuICAgIClcbiAgXSlcbn07XG4iXX0=
@@ -0,0 +1,37 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ /** Injection token that can be used to access the data that was passed in to a snack bar. */
3
+ export const BAO_SNACK_BAR_DATA = new InjectionToken('BaoSnackBarData');
4
+ /**
5
+ * Configuration used when opening a snack-bar.
6
+ */
7
+ export class BaoSnackBarConfig {
8
+ constructor() {
9
+ /** The message to display in the snackbar. */
10
+ this.message = 'No message';
11
+ /** The type of snackbar template to display. */
12
+ this.toastType = 'info';
13
+ /**
14
+ * The attached action to the snack bar. If the name of the action matches an icon provided as part of
15
+ * angular-ui icon dictionnary an icon will be displayed instead of text.
16
+ * */
17
+ this.actionLabelOrIcon = '';
18
+ /** Displays the close button when set to true */
19
+ this.showClose = false;
20
+ /** The length of time in milliseconds to wait before automatically dismissing the snack bar. */
21
+ this.duration = 5000;
22
+ /** The politeness level for the MatAriaLiveAnnouncer announcement. */
23
+ this.politeness = 'assertive';
24
+ /**
25
+ * Message to be announced by the LiveAnnouncer. When opening a snackbar without a custom
26
+ * component or template, the announcement message will default to the specified message.
27
+ */
28
+ this.announcementMessage = '';
29
+ /** Data being injected into the child component. */
30
+ this.data = null;
31
+ /** The horizontal position to place the snack bar. */
32
+ this.horizontalPosition = 'left';
33
+ /** The vertical position to place the snack bar. */
34
+ this.verticalPosition = 'bottom';
35
+ }
36
+ }
37
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic25hY2stYmFyLWNvbmZpZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2FuZ3VsYXItdWkvc3JjL2xpYi9zbmFjay1iYXIvc25hY2stYmFyLWNvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFPQSxPQUFPLEVBQUUsY0FBYyxFQUFvQixNQUFNLGVBQWUsQ0FBQztBQUVqRSw2RkFBNkY7QUFDN0YsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsSUFBSSxjQUFjLENBQU0saUJBQWlCLENBQUMsQ0FBQztBQWdCN0U7O0dBRUc7QUFDSCxNQUFNLE9BQU8saUJBQWlCO0lBQTlCO1FBQ0UsOENBQThDO1FBQ3ZDLFlBQU8sR0FBRyxZQUFZLENBQUM7UUFFOUIsZ0RBQWdEO1FBQ3pDLGNBQVMsR0FBMEIsTUFBTSxDQUFDO1FBRWpEOzs7YUFHSztRQUNFLHNCQUFpQixHQUFZLEVBQUUsQ0FBQztRQUV2QyxpREFBaUQ7UUFDMUMsY0FBUyxHQUFhLEtBQUssQ0FBQztRQUVuQyxnR0FBZ0c7UUFDekYsYUFBUSxHQUFZLElBQUksQ0FBQztRQUVoQyxzRUFBc0U7UUFDL0QsZUFBVSxHQUF3QixXQUFXLENBQUM7UUFFckQ7OztXQUdHO1FBQ0ksd0JBQW1CLEdBQVksRUFBRSxDQUFDO1FBY3pDLG9EQUFvRDtRQUM3QyxTQUFJLEdBQWMsSUFBSSxDQUFDO1FBRTlCLHNEQUFzRDtRQUMvQyx1QkFBa0IsR0FBbUMsTUFBTSxDQUFDO1FBRW5FLG9EQUFvRDtRQUM3QyxxQkFBZ0IsR0FBaUMsUUFBUSxDQUFDO0lBQ25FLENBQUM7Q0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjMgVmlsbGUgZGUgTW9udHJlYWwuIEFsbCByaWdodHMgcmVzZXJ2ZWQuXG4gKiBMaWNlbnNlZCB1bmRlciB0aGUgTUlUIGxpY2Vuc2UuXG4gKiBTZWUgTElDRU5TRSBmaWxlIGluIHRoZSBwcm9qZWN0IHJvb3QgZm9yIGZ1bGwgbGljZW5zZSBpbmZvcm1hdGlvbi5cbiAqL1xuaW1wb3J0IHsgQXJpYUxpdmVQb2xpdGVuZXNzIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL2ExMXknO1xuaW1wb3J0IHsgRGlyZWN0aW9uIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL2JpZGknO1xuaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4sIFZpZXdDb250YWluZXJSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuLyoqIEluamVjdGlvbiB0b2tlbiB0aGF0IGNhbiBiZSB1c2VkIHRvIGFjY2VzcyB0aGUgZGF0YSB0aGF0IHdhcyBwYXNzZWQgaW4gdG8gYSBzbmFjayBiYXIuICovXG5leHBvcnQgY29uc3QgQkFPX1NOQUNLX0JBUl9EQVRBID0gbmV3IEluamVjdGlvblRva2VuPGFueT4oJ0Jhb1NuYWNrQmFyRGF0YScpO1xuXG4vKiogUG9zc2libGUgdmFsdWVzIGZvciBob3Jpem9udGFsUG9zaXRpb24gb24gTWF0U25hY2tCYXJDb25maWcuICovXG5leHBvcnQgdHlwZSBCYW9TbmFja0Jhckhvcml6b250YWxQb3NpdGlvbiA9XG4gIHwgJ3N0YXJ0J1xuICB8ICdjZW50ZXInXG4gIHwgJ2VuZCdcbiAgfCAnbGVmdCdcbiAgfCAncmlnaHQnO1xuXG4vKiogUG9zc2libGUgdmFsdWVzIGZvciB2ZXJ0aWNhbFBvc2l0aW9uIG9uIE1hdFNuYWNrQmFyQ29uZmlnLiAqL1xuZXhwb3J0IHR5cGUgQmFvU25hY2tCYXJWZXJ0aWNhbFBvc2l0aW9uID0gJ3RvcCcgfCAnYm90dG9tJztcblxuLyoqIFBvc3NpYmxlIHR5cGVzIG9mIHRvYXN0IHRvIGRpc3BsYXkgdGhlIHNuYWNrIGJhciAqL1xuZXhwb3J0IHR5cGUgQmFvU25hY2tCYXJUb2FzdFR5cGUgPSAnaW5mbycgfCAnc3VjY2VzcycgfCAnZGFuZ2VyJztcblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIHVzZWQgd2hlbiBvcGVuaW5nIGEgc25hY2stYmFyLlxuICovXG5leHBvcnQgY2xhc3MgQmFvU25hY2tCYXJDb25maWc8RCA9IGFueT4ge1xuICAvKiogVGhlIG1lc3NhZ2UgdG8gZGlzcGxheSBpbiB0aGUgc25hY2tiYXIuICovXG4gIHB1YmxpYyBtZXNzYWdlID0gJ05vIG1lc3NhZ2UnO1xuXG4gIC8qKiBUaGUgdHlwZSBvZiBzbmFja2JhciB0ZW1wbGF0ZSB0byBkaXNwbGF5LiAqL1xuICBwdWJsaWMgdG9hc3RUeXBlPzogQmFvU25hY2tCYXJUb2FzdFR5cGUgPSAnaW5mbyc7XG5cbiAgLyoqXG4gICAqIFRoZSBhdHRhY2hlZCBhY3Rpb24gdG8gdGhlIHNuYWNrIGJhci4gSWYgdGhlIG5hbWUgb2YgdGhlIGFjdGlvbiBtYXRjaGVzIGFuIGljb24gcHJvdmlkZWQgYXMgcGFydCBvZlxuICAgKiBhbmd1bGFyLXVpIGljb24gZGljdGlvbm5hcnkgYW4gaWNvbiB3aWxsIGJlIGRpc3BsYXllZCBpbnN0ZWFkIG9mIHRleHQuXG4gICAqICovXG4gIHB1YmxpYyBhY3Rpb25MYWJlbE9ySWNvbj86IHN0cmluZyA9ICcnO1xuXG4gIC8qKiBEaXNwbGF5cyB0aGUgY2xvc2UgYnV0dG9uIHdoZW4gc2V0IHRvIHRydWUgKi9cbiAgcHVibGljIHNob3dDbG9zZT86IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKiogVGhlIGxlbmd0aCBvZiB0aW1lIGluIG1pbGxpc2Vjb25kcyB0byB3YWl0IGJlZm9yZSBhdXRvbWF0aWNhbGx5IGRpc21pc3NpbmcgdGhlIHNuYWNrIGJhci4gKi9cbiAgcHVibGljIGR1cmF0aW9uPzogbnVtYmVyID0gNTAwMDtcblxuICAvKiogVGhlIHBvbGl0ZW5lc3MgbGV2ZWwgZm9yIHRoZSBNYXRBcmlhTGl2ZUFubm91bmNlciBhbm5vdW5jZW1lbnQuICovXG4gIHB1YmxpYyBwb2xpdGVuZXNzPzogQXJpYUxpdmVQb2xpdGVuZXNzID0gJ2Fzc2VydGl2ZSc7XG5cbiAgLyoqXG4gICAqIE1lc3NhZ2UgdG8gYmUgYW5ub3VuY2VkIGJ5IHRoZSBMaXZlQW5ub3VuY2VyLiBXaGVuIG9wZW5pbmcgYSBzbmFja2JhciB3aXRob3V0IGEgY3VzdG9tXG4gICAqIGNvbXBvbmVudCBvciB0ZW1wbGF0ZSwgdGhlIGFubm91bmNlbWVudCBtZXNzYWdlIHdpbGwgZGVmYXVsdCB0byB0aGUgc3BlY2lmaWVkIG1lc3NhZ2UuXG4gICAqL1xuICBwdWJsaWMgYW5ub3VuY2VtZW50TWVzc2FnZT86IHN0cmluZyA9ICcnO1xuXG4gIC8qKlxuICAgKiBUaGUgdmlldyBjb250YWluZXIgdGhhdCBzZXJ2ZXMgYXMgdGhlIHBhcmVudCBmb3IgdGhlIHNuYWNrYmFyIGZvciB0aGUgcHVycG9zZXMgb2YgZGVwZW5kZW5jeVxuICAgKiBpbmplY3Rpb24uIE5vdGU6IHRoaXMgZG9lcyBub3QgYWZmZWN0IHdoZXJlIHRoZSBzbmFja2JhciBpcyBpbnNlcnRlZCBpbiB0aGUgRE9NLlxuICAgKi9cbiAgcHVibGljIHZpZXdDb250YWluZXJSZWY/OiBWaWV3Q29udGFpbmVyUmVmO1xuXG4gIC8qKiBFeHRyYSBDU1MgY2xhc3NlcyB0byBiZSBhZGRlZCB0byB0aGUgc25hY2sgYmFyIGNvbnRhaW5lci4gKi9cbiAgcHVibGljIHBhbmVsQ2xhc3M/OiBzdHJpbmcgfCBzdHJpbmdbXTtcblxuICAvKiogVGV4dCBsYXlvdXQgZGlyZWN0aW9uIGZvciB0aGUgc25hY2sgYmFyLiAqL1xuICBwdWJsaWMgZGlyZWN0aW9uPzogRGlyZWN0aW9uO1xuXG4gIC8qKiBEYXRhIGJlaW5nIGluamVjdGVkIGludG8gdGhlIGNoaWxkIGNvbXBvbmVudC4gKi9cbiAgcHVibGljIGRhdGE/OiBEIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIFRoZSBob3Jpem9udGFsIHBvc2l0aW9uIHRvIHBsYWNlIHRoZSBzbmFjayBiYXIuICovXG4gIHB1YmxpYyBob3Jpem9udGFsUG9zaXRpb24/OiBCYW9TbmFja0Jhckhvcml6b250YWxQb3NpdGlvbiA9ICdsZWZ0JztcblxuICAvKiogVGhlIHZlcnRpY2FsIHBvc2l0aW9uIHRvIHBsYWNlIHRoZSBzbmFjayBiYXIuICovXG4gIHB1YmxpYyB2ZXJ0aWNhbFBvc2l0aW9uPzogQmFvU25hY2tCYXJWZXJ0aWNhbFBvc2l0aW9uID0gJ2JvdHRvbSc7XG59XG4iXX0=
@@ -0,0 +1,272 @@
1
+ import { BasePortalOutlet, CdkPortalOutlet } from '@angular/cdk/portal';
2
+ import { DOCUMENT } from '@angular/common';
3
+ import { ChangeDetectionStrategy, Component, inject, ViewChild, ViewEncapsulation } from '@angular/core';
4
+ import { Subject } from 'rxjs';
5
+ import { take } from 'rxjs/operators';
6
+ import { matSnackBarAnimations } from './snack-bar-animations';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/cdk/platform";
9
+ import * as i2 from "./snack-bar-config";
10
+ import * as i3 from "@angular/cdk/portal";
11
+ let uniqueId = 0;
12
+ /**
13
+ * Internal component that wraps user-provided snack bar content.
14
+ * @docs-private
15
+ */
16
+ export class BaoSnackBarContainerComponent extends BasePortalOutlet {
17
+ constructor(_ngZone, _elementRef, _changeDetectorRef, _platform,
18
+ /** The snack bar configuration. */
19
+ snackBarConfig) {
20
+ super();
21
+ this._ngZone = _ngZone;
22
+ this._elementRef = _elementRef;
23
+ this._changeDetectorRef = _changeDetectorRef;
24
+ this._platform = _platform;
25
+ this.snackBarConfig = snackBarConfig;
26
+ this._document = inject(DOCUMENT);
27
+ this._trackedModals = new Set();
28
+ /** Subject for notifying that the snack bar has announced to screen readers. */
29
+ this._onAnnounce = new Subject();
30
+ /** Subject for notifying that the snack bar has exited from view. */
31
+ this._onExit = new Subject();
32
+ /** Subject for notifying that the snack bar has finished entering the view. */
33
+ this._onEnter = new Subject();
34
+ /** The state of the snack bar animations. */
35
+ this._animationState = 'void';
36
+ /** The number of milliseconds to wait before announcing the snack bar's content. */
37
+ this._announceDelay = 150;
38
+ /** Whether the component has been destroyed. */
39
+ this._destroyed = false;
40
+ /** Unique ID of the aria-live element. */
41
+ this._liveElementId = `bao-snack-bar-container-live-${uniqueId++}`;
42
+ /**
43
+ * Attaches a DOM portal to the snack bar container.
44
+ * @deprecated To be turned into a method.
45
+ * @breaking-change 10.0.0
46
+ */
47
+ this.attachDomPortal = (portal) => {
48
+ this.assertNotAttached();
49
+ const result = this._portalOutlet.attachDomPortal(portal);
50
+ this._afterPortalAttached();
51
+ return result;
52
+ };
53
+ // Use aria-live rather than a live role like 'alert' or 'status'
54
+ // because NVDA and JAWS have show inconsistent behavior with live roles.
55
+ if (snackBarConfig.politeness === 'assertive' &&
56
+ !snackBarConfig.announcementMessage) {
57
+ this._live = 'assertive';
58
+ }
59
+ else if (snackBarConfig.politeness === 'off') {
60
+ this._live = 'off';
61
+ }
62
+ else {
63
+ this._live = 'polite';
64
+ }
65
+ // Only set role for Firefox. Set role based on aria-live because setting role="alert" implies
66
+ // aria-live="assertive" which may cause issues if aria-live is set to "polite" above.
67
+ if (this._platform.FIREFOX) {
68
+ if (this._live === 'polite') {
69
+ this._role = 'status';
70
+ }
71
+ if (this._live === 'assertive') {
72
+ this._role = 'alert';
73
+ }
74
+ }
75
+ }
76
+ /** Attach a component portal as content to this snack bar container. */
77
+ attachComponentPortal(portal) {
78
+ this.assertNotAttached();
79
+ this.applySnackBarClasses();
80
+ const result = this._portalOutlet.attachComponentPortal(portal);
81
+ this._afterPortalAttached();
82
+ return result;
83
+ }
84
+ /** Attach a template portal as content to this snack bar container. */
85
+ attachTemplatePortal(portal) {
86
+ this.assertNotAttached();
87
+ this.applySnackBarClasses();
88
+ const result = this._portalOutlet.attachTemplatePortal(portal);
89
+ this._afterPortalAttached();
90
+ return result;
91
+ }
92
+ /** Handle end of animations, updating the state of the snackbar. */
93
+ onAnimationEnd(event) {
94
+ const { fromState, toState } = event;
95
+ if ((toState === 'void' && fromState !== 'void') || toState === 'hidden') {
96
+ this.completeExit();
97
+ }
98
+ if (toState === 'visible') {
99
+ // Note: we shouldn't use `this` inside the zone callback,
100
+ // because it can cause a memory leak.
101
+ const onEnter = this._onEnter;
102
+ this._ngZone.run(() => {
103
+ onEnter.next();
104
+ onEnter.complete();
105
+ });
106
+ }
107
+ }
108
+ /** Begin animation of snack bar entrance into view. */
109
+ enter() {
110
+ if (!this._destroyed) {
111
+ this._animationState = 'visible';
112
+ this._changeDetectorRef.detectChanges();
113
+ this.screenReaderAnnounce();
114
+ }
115
+ }
116
+ /** Begin animation of the snack bar exiting from view. */
117
+ exit() {
118
+ // Note: this one transitions to `hidden`, rather than `void`, in order to handle the case
119
+ // where multiple snack bars are opened in quick succession (e.g. two consecutive calls to
120
+ // `MatSnackBar.open`).
121
+ this._animationState = 'hidden';
122
+ // Mark this element with an 'exit' attribute to indicate that the snackbar has
123
+ // been dismissed and will soon be removed from the DOM. This is used by the snackbar
124
+ // test harness.
125
+ this._elementRef.nativeElement.setAttribute('mat-exit', '');
126
+ // If the snack bar hasn't been announced by the time it exits it wouldn't have been open
127
+ // long enough to visually read it either, so clear the timeout for announcing.
128
+ clearTimeout(this._announceTimeoutId);
129
+ return this._onExit;
130
+ }
131
+ /** Makes sure the exit callbacks have been invoked when the element is destroyed. */
132
+ ngOnDestroy() {
133
+ this._destroyed = true;
134
+ this.completeExit();
135
+ }
136
+ /**
137
+ * Waits for the zone to settle before removing the element. Helps prevent
138
+ * errors where we end up removing an element which is in the middle of an animation.
139
+ */
140
+ completeExit() {
141
+ this._ngZone.onMicrotaskEmpty.pipe(take(1)).subscribe(() => {
142
+ this._onExit.next();
143
+ this._onExit.complete();
144
+ });
145
+ }
146
+ /** Applies the various positioning and user-configured CSS classes to the snack bar. */
147
+ applySnackBarClasses() {
148
+ const element = this._elementRef.nativeElement;
149
+ const panelClasses = this.snackBarConfig.panelClass;
150
+ if (panelClasses) {
151
+ if (Array.isArray(panelClasses)) {
152
+ // Note that we can't use a spread here, because IE doesn't support multiple arguments.
153
+ panelClasses.forEach(cssClass => element.classList.add(cssClass));
154
+ }
155
+ else {
156
+ element.classList.add(panelClasses);
157
+ }
158
+ }
159
+ if (this.snackBarConfig.horizontalPosition === 'center') {
160
+ element.classList.add('bao-snack-bar-center');
161
+ }
162
+ if (this.snackBarConfig.verticalPosition === 'top') {
163
+ element.classList.add('bao-snack-bar-top');
164
+ }
165
+ }
166
+ /**
167
+ * Called after the portal contents have been attached. Can be
168
+ * used to modify the DOM once it's guaranteed to be in place.
169
+ */
170
+ _afterPortalAttached() {
171
+ const element = this._elementRef.nativeElement;
172
+ const panelClasses = this.snackBarConfig.panelClass;
173
+ if (panelClasses) {
174
+ if (Array.isArray(panelClasses)) {
175
+ // Note that we can't use a spread here, because IE doesn't support multiple arguments.
176
+ panelClasses.forEach(cssClass => element.classList.add(cssClass));
177
+ }
178
+ else {
179
+ element.classList.add(panelClasses);
180
+ }
181
+ }
182
+ this._exposeToModals();
183
+ }
184
+ /**
185
+ * Some browsers won't expose the accessibility node of the live element if there is an
186
+ * `aria-modal` and the live element is outside of it. This method works around the issue by
187
+ * pointing the `aria-owns` of all modals to the live element.
188
+ */
189
+ _exposeToModals() {
190
+ // Note that the selector here is limited to CDK overlays at the moment in order to reduce the
191
+ // section of the DOM we need to look through. This should cover all the cases we support, but
192
+ // the selector can be expanded if it turns out to be too narrow.
193
+ const id = this._liveElementId;
194
+ const modals = this._document.querySelectorAll('body > .cdk-overlay-container [aria-modal="true"]');
195
+ for (let i = 0; i < modals.length; i++) {
196
+ const modal = modals[i];
197
+ const ariaOwns = modal.getAttribute('aria-owns');
198
+ this._trackedModals.add(modal);
199
+ if (!ariaOwns) {
200
+ modal.setAttribute('aria-owns', id);
201
+ }
202
+ else if (ariaOwns.indexOf(id) === -1) {
203
+ modal.setAttribute('aria-owns', ariaOwns + ' ' + id);
204
+ }
205
+ }
206
+ }
207
+ /** Clears the references to the live element from any modals it was added to. */
208
+ _clearFromModals() {
209
+ this._trackedModals.forEach(modal => {
210
+ const ariaOwns = modal.getAttribute('aria-owns');
211
+ if (ariaOwns) {
212
+ const newValue = ariaOwns.replace(this._liveElementId, '').trim();
213
+ if (newValue.length > 0) {
214
+ modal.setAttribute('aria-owns', newValue);
215
+ }
216
+ else {
217
+ modal.removeAttribute('aria-owns');
218
+ }
219
+ }
220
+ });
221
+ this._trackedModals.clear();
222
+ }
223
+ /** Asserts that no content is already attached to the container. */
224
+ assertNotAttached() {
225
+ if (this._portalOutlet.hasAttached()) {
226
+ throw Error('Attempting to attach snack bar content after content is already attached');
227
+ }
228
+ }
229
+ /**
230
+ * Starts a timeout to move the snack bar content to the live region so screen readers will
231
+ * announce it.
232
+ */
233
+ screenReaderAnnounce() {
234
+ if (!this._announceTimeoutId) {
235
+ this._ngZone.runOutsideAngular(() => {
236
+ this._announceTimeoutId = window.setTimeout(() => {
237
+ const inertElement = this._elementRef.nativeElement.querySelector('[aria-hidden]');
238
+ const liveElement = this._elementRef.nativeElement.querySelector('[aria-live]');
239
+ if (inertElement && liveElement) {
240
+ // If an element in the snack bar content is focused before being moved
241
+ // track it and restore focus after moving to the live region.
242
+ let focusedElement = null;
243
+ if (this._platform.isBrowser &&
244
+ document.activeElement instanceof HTMLElement &&
245
+ inertElement.contains(document.activeElement)) {
246
+ focusedElement = document.activeElement;
247
+ }
248
+ inertElement.removeAttribute('aria-hidden');
249
+ liveElement.appendChild(inertElement);
250
+ focusedElement?.focus();
251
+ this._onAnnounce.next();
252
+ this._onAnnounce.complete();
253
+ }
254
+ }, this._announceDelay);
255
+ });
256
+ }
257
+ }
258
+ }
259
+ BaoSnackBarContainerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoSnackBarContainerComponent, deps: [{ token: i0.NgZone }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: i1.Platform }, { token: i2.BaoSnackBarConfig }], target: i0.ɵɵFactoryTarget.Component });
260
+ BaoSnackBarContainerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: BaoSnackBarContainerComponent, selector: "bao-snack-bar-container", host: { listeners: { "@state.done": "onAnimationEnd($event)" }, properties: { "@state": "_animationState" }, classAttribute: "bao-snack-bar-container" }, viewQueries: [{ propertyName: "_portalOutlet", first: true, predicate: CdkPortalOutlet, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<!-- Initialy 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\n [attr.aria-live]=\"_live\"\n [attr.role]=\"_role\"\n [attr.id]=\"_liveElementId\"\n></div>\n", styles: [".bao-container{padding-right:16px;padding-left:16px;margin-right:auto;margin-left:auto;width:100%}@media (min-width: 576px){.bao-container{max-width:576px}}@media (min-width: 768px){.bao-container{max-width:768px}}@media (min-width: 992px){.bao-container{max-width:992px}}@media (min-width: 1200px){.bao-container{max-width:1200px}}.bao-row{display:flex;flex-wrap:wrap;margin-right:-16px;margin-left:-16px}.bao-col-12,.bao-col-lg-7{position:relative;width:100%;padding-right:1rem;padding-left:1rem}@media (min-width: 992px){.bao-col-lg-7{flex:0 0 58.33333%;max-width:58.33333%}}.cdk-overlay-pane{width:100%}@media (min-width: 768px){.cdk-overlay-pane{margin:0;width:auto}}.bao-snack-bar-container{border-radius:.25rem;box-sizing:border-box;display:block;margin:1rem;max-width:100vw;min-width:15rem;min-height:3rem;transform-origin:center;width:100%}\n"], directives: [{ type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }], animations: [matSnackBarAnimations.snackBarState], changeDetection: i0.ChangeDetectionStrategy.Default, encapsulation: i0.ViewEncapsulation.None });
261
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: BaoSnackBarContainerComponent, decorators: [{
262
+ type: Component,
263
+ args: [{ selector: 'bao-snack-bar-container', changeDetection: ChangeDetectionStrategy.Default, encapsulation: ViewEncapsulation.None, animations: [matSnackBarAnimations.snackBarState], host: {
264
+ class: 'bao-snack-bar-container',
265
+ '[@state]': '_animationState',
266
+ '(@state.done)': 'onAnimationEnd($event)'
267
+ }, template: "<!-- Initialy 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\n [attr.aria-live]=\"_live\"\n [attr.role]=\"_role\"\n [attr.id]=\"_liveElementId\"\n></div>\n", styles: [".bao-container{padding-right:16px;padding-left:16px;margin-right:auto;margin-left:auto;width:100%}@media (min-width: 576px){.bao-container{max-width:576px}}@media (min-width: 768px){.bao-container{max-width:768px}}@media (min-width: 992px){.bao-container{max-width:992px}}@media (min-width: 1200px){.bao-container{max-width:1200px}}.bao-row{display:flex;flex-wrap:wrap;margin-right:-16px;margin-left:-16px}.bao-col-12,.bao-col-lg-7{position:relative;width:100%;padding-right:1rem;padding-left:1rem}@media (min-width: 992px){.bao-col-lg-7{flex:0 0 58.33333%;max-width:58.33333%}}.cdk-overlay-pane{width:100%}@media (min-width: 768px){.cdk-overlay-pane{margin:0;width:auto}}.bao-snack-bar-container{border-radius:.25rem;box-sizing:border-box;display:block;margin:1rem;max-width:100vw;min-width:15rem;min-height:3rem;transform-origin:center;width:100%}\n"] }]
268
+ }], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: i1.Platform }, { type: i2.BaoSnackBarConfig }]; }, propDecorators: { _portalOutlet: [{
269
+ type: ViewChild,
270
+ args: [CdkPortalOutlet, { static: true }]
271
+ }] } });
272
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snack-bar-container.js","sourceRoot":"","sources":["../../../../../projects/angular-ui/src/lib/snack-bar/snack-bar-container.ts","../../../../../projects/angular-ui/src/lib/snack-bar/snack-bar-container.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023 Ville de Montreal. All rights reserved.\n * Licensed under the MIT license.\n * See LICENSE file in the project root for full license information.\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  DomPortal,\n  TemplatePortal\n} from '@angular/cdk/portal';\nimport { DOCUMENT } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ComponentRef,\n  ElementRef,\n  EmbeddedViewRef,\n  inject,\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 { BaoSnackBarConfig } from './snack-bar-config';\n\nlet uniqueId = 0;\n\n/**\n * Internal interface for a snack bar container.\n * @docs-private\n */\nexport interface IBaoSnackBarContainer {\n  snackBarConfig: BaoSnackBarConfig;\n  _onAnnounce: Subject<any>;\n  _onExit: Subject<any>;\n  _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: 'bao-snack-bar-container',\n  templateUrl: 'snack-bar-container.html',\n  styleUrls: ['snack-bar-container.scss'],\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  // eslint-disable-next-line\n  changeDetection: ChangeDetectionStrategy.Default,\n  encapsulation: ViewEncapsulation.None,\n  animations: [matSnackBarAnimations.snackBarState],\n  host: {\n    class: 'bao-snack-bar-container',\n    '[@state]': '_animationState',\n    '(@state.done)': 'onAnimationEnd($event)'\n  }\n})\nexport class BaoSnackBarContainerComponent\n  extends BasePortalOutlet\n  implements OnDestroy, IBaoSnackBarContainer\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  private _document = inject(DOCUMENT);\n  private _trackedModals = new Set<Element>();\n\n  /** Subject for notifying that the snack bar has announced to screen readers. */\n  public readonly _onAnnounce: Subject<void> = new Subject();\n\n  /** Subject for notifying that the snack bar has exited from view. */\n  public readonly _onExit: Subject<void> = new Subject();\n\n  /** Subject for notifying that the snack bar has finished entering the view. */\n  public readonly _onEnter: Subject<void> = new Subject();\n\n  /** The state of the snack bar animations. */\n  public _animationState = 'void';\n\n  /** aria-live value for the live region. */\n  public _live: AriaLivePoliteness;\n\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  /**\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  /** Unique ID of the aria-live element. */\n  readonly _liveElementId = `bao-snack-bar-container-live-${uniqueId++}`;\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: BaoSnackBarConfig\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 (\n      snackBarConfig.politeness === 'assertive' &&\n      !snackBarConfig.announcementMessage\n    ) {\n      this._live = 'assertive';\n    } else if (snackBarConfig.politeness === 'off') {\n      this._live = 'off';\n    } else {\n      this._live = 'polite';\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  public attachComponentPortal<T>(portal: ComponentPortal<T>): ComponentRef<T> {\n    this.assertNotAttached();\n    this.applySnackBarClasses();\n    const result = this._portalOutlet.attachComponentPortal(portal);\n    this._afterPortalAttached();\n    return result;\n  }\n\n  /** Attach a template portal as content to this snack bar container. */\n  public attachTemplatePortal<C>(\n    portal: TemplatePortal<C>\n  ): EmbeddedViewRef<C> {\n    this.assertNotAttached();\n    this.applySnackBarClasses();\n    const result = this._portalOutlet.attachTemplatePortal(portal);\n    this._afterPortalAttached();\n    return result;\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  public attachDomPortal = (portal: DomPortal) => {\n    this.assertNotAttached();\n    const result = this._portalOutlet.attachDomPortal(portal);\n    this._afterPortalAttached();\n    return result;\n  };\n\n  /** Handle end of animations, updating the state of the snackbar. */\n  public 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  public 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  public 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  public 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('bao-snack-bar-center');\n    }\n\n    if (this.snackBarConfig.verticalPosition === 'top') {\n      element.classList.add('bao-snack-bar-top');\n    }\n  }\n\n  /**\n   * Called after the portal contents have been attached. Can be\n   * used to modify the DOM once it's guaranteed to be in place.\n   */\n  protected _afterPortalAttached() {\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    this._exposeToModals();\n  }\n\n  /**\n   * Some browsers won't expose the accessibility node of the live element if there is an\n   * `aria-modal` and the live element is outside of it. This method works around the issue by\n   * pointing the `aria-owns` of all modals to the live element.\n   */\n  private _exposeToModals() {\n    // Note that the selector here is limited to CDK overlays at the moment in order to reduce the\n    // section of the DOM we need to look through. This should cover all the cases we support, but\n    // the selector can be expanded if it turns out to be too narrow.\n    const id = this._liveElementId;\n    const modals = this._document.querySelectorAll(\n      'body > .cdk-overlay-container [aria-modal=\"true\"]'\n    );\n\n    for (let i = 0; i < modals.length; i++) {\n      const modal = modals[i];\n      const ariaOwns = modal.getAttribute('aria-owns');\n      this._trackedModals.add(modal);\n\n      if (!ariaOwns) {\n        modal.setAttribute('aria-owns', id);\n      } else if (ariaOwns.indexOf(id) === -1) {\n        modal.setAttribute('aria-owns', ariaOwns + ' ' + id);\n      }\n    }\n  }\n\n  /** Clears the references to the live element from any modals it was added to. */\n  private _clearFromModals() {\n    this._trackedModals.forEach(modal => {\n      const ariaOwns = modal.getAttribute('aria-owns');\n\n      if (ariaOwns) {\n        const newValue = ariaOwns.replace(this._liveElementId, '').trim();\n\n        if (newValue.length > 0) {\n          modal.setAttribute('aria-owns', newValue);\n        } else {\n          modal.removeAttribute('aria-owns');\n        }\n      }\n    });\n    this._trackedModals.clear();\n  }\n\n  /** Asserts that no content is already attached to the container. */\n  private assertNotAttached() {\n    if (this._portalOutlet.hasAttached()) {\n      throw Error(\n        'Attempting to attach snack bar content after content is already attached'\n      );\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 = window.setTimeout(() => {\n          const inertElement =\n            this._elementRef.nativeElement.querySelector('[aria-hidden]');\n          const liveElement =\n            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 (\n              this._platform.isBrowser &&\n              document.activeElement instanceof HTMLElement &&\n              inertElement.contains(document.activeElement)\n            ) {\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","<!-- Initialy 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\n  [attr.aria-live]=\"_live\"\n  [attr.role]=\"_role\"\n  [attr.id]=\"_liveElementId\"\n></div>\n"]}
@@ -0,0 +1,75 @@
1
+ import { Subject } from 'rxjs';
2
+ /** Maximum amount of milliseconds that can be passed into setTimeout. */
3
+ const MAX_TIMEOUT = Math.pow(2, 31) - 1;
4
+ /**
5
+ * Reference to a snack bar dispatched from the snack bar service.
6
+ */
7
+ export class BaoSnackBarRef {
8
+ constructor(containerInstance, _overlayRef) {
9
+ this._overlayRef = _overlayRef;
10
+ /** Subject for notifying the user that the snack bar has been dismissed. */
11
+ this._afterDismissed = new Subject();
12
+ /** Subject for notifying the user that the snack bar has opened and appeared. */
13
+ this._afterOpened = new Subject();
14
+ /** Subject for notifying the user that the snack bar action was called. */
15
+ this._onAction = new Subject();
16
+ /** Whether the snack bar was dismissed using the action button. */
17
+ this._dismissedByAction = false;
18
+ this.containerInstance = containerInstance;
19
+ // Dismiss snackbar on action.
20
+ this.onAction().subscribe(() => this.dismiss());
21
+ containerInstance._onExit.subscribe(() => this.finishDismiss());
22
+ }
23
+ /** Dismisses the snack bar. */
24
+ dismiss() {
25
+ if (!this._afterDismissed.closed) {
26
+ this.containerInstance.exit();
27
+ }
28
+ clearTimeout(this._durationTimeoutId);
29
+ }
30
+ /** Marks the snackbar action clicked. */
31
+ dismissWithAction() {
32
+ if (!this._onAction.closed) {
33
+ this._dismissedByAction = true;
34
+ this._onAction.next();
35
+ this._onAction.complete();
36
+ }
37
+ }
38
+ /** Dismisses the snack bar after some duration */
39
+ dismissAfter(duration) {
40
+ // Note that we need to cap the duration to the maximum value for setTimeout, because
41
+ // it'll revert to 1 if somebody passes in something greater (e.g. `Infinity`). See #17234.
42
+ // @TODO: window.setTimeout() ?
43
+ this._durationTimeoutId = window.setTimeout(() => this.dismiss(), Math.min(duration, MAX_TIMEOUT));
44
+ }
45
+ /** Marks the snackbar as opened */
46
+ open() {
47
+ if (!this._afterOpened.closed) {
48
+ this._afterOpened.next();
49
+ this._afterOpened.complete();
50
+ }
51
+ }
52
+ /** Gets an observable that is notified when the snack bar is finished closing. */
53
+ afterDismissed() {
54
+ return this._afterDismissed;
55
+ }
56
+ /** Gets an observable that is notified when the snack bar has opened and appeared. */
57
+ afterOpened() {
58
+ return this.containerInstance._onEnter;
59
+ }
60
+ /** Gets an observable that is notified when the snack bar action is called. */
61
+ onAction() {
62
+ return this._onAction;
63
+ }
64
+ /** Cleans up the DOM after closing. */
65
+ finishDismiss() {
66
+ this._overlayRef.dispose();
67
+ if (!this._onAction.closed) {
68
+ this._onAction.complete();
69
+ }
70
+ this._afterDismissed.next({ dismissedByAction: this._dismissedByAction });
71
+ this._afterDismissed.complete();
72
+ this._dismissedByAction = false;
73
+ }
74
+ }
75
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snack-bar-ref.js","sourceRoot":"","sources":["../../../../../projects/angular-ui/src/lib/snack-bar/snack-bar-ref.ts"],"names":[],"mappings":"AAMA,OAAO,EAAc,OAAO,EAAE,MAAM,MAAM,CAAC;AAS3C,yEAAyE;AACzE,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;AAExC;;GAEG;AACH,MAAM,OAAO,cAAc;IA4BzB,YACE,iBAAwC,EAChC,WAAuB;QAAvB,gBAAW,GAAX,WAAW,CAAY;QApBjC,4EAA4E;QAC3D,oBAAe,GAAG,IAAI,OAAO,EAAuB,CAAC;QAEtE,iFAAiF;QAChE,iBAAY,GAAG,IAAI,OAAO,EAAQ,CAAC;QAEpD,2EAA2E;QAC1D,cAAS,GAAG,IAAI,OAAO,EAAQ,CAAC;QAQjD,mEAAmE;QAC3D,uBAAkB,GAAG,KAAK,CAAC;QAMjC,IAAI,CAAC,iBAAiB,GAAG,iBAAiB,CAAC;QAC3C,8BAA8B;QAC9B,IAAI,CAAC,QAAQ,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAChD,iBAAiB,CAAC,OAAO,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC;IAClE,CAAC;IAED,+BAA+B;IACxB,OAAO;QACZ,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE;YAChC,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,CAAC;SAC/B;QACD,YAAY,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;IACxC,CAAC;IAED,yCAAyC;IAClC,iBAAiB;QACtB,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE;YAC1B,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;YAC/B,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;YACtB,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;SAC3B;IACH,CAAC;IAED,kDAAkD;IAC3C,YAAY,CAAC,QAAgB;QAClC,qFAAqF;QACrF,2FAA2F;QAE3F,+BAA+B;QAC/B,IAAI,CAAC,kBAAkB,GAAG,MAAM,CAAC,UAAU,CACzC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,EACpB,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE,WAAW,CAAC,CAChC,CAAC;IACJ,CAAC;IAED,mCAAmC;IAC5B,IAAI;QACT,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE;YAC7B,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC;YACzB,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC;SAC9B;IACH,CAAC;IAED,kFAAkF;IAC3E,cAAc;QACnB,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED,sFAAsF;IAC/E,WAAW;QAChB,OAAO,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC;IACzC,CAAC;IAED,+EAA+E;IACxE,QAAQ;QACb,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IAED,uCAAuC;IAC/B,aAAa;QACnB,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;QAE3B,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE;YAC1B,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;SAC3B;QAED,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,iBAAiB,EAAE,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC;QAC1E,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,CAAC;QAChC,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;IAClC,CAAC;CACF","sourcesContent":["/*\n * Copyright (c) 2023 Ville de Montreal. All rights reserved.\n * Licensed under the MIT license.\n * See LICENSE file in the project root for full license information.\n */\nimport { OverlayRef } from '@angular/cdk/overlay';\nimport { Observable, Subject } from 'rxjs';\nimport { IBaoSnackBarContainer } from './snack-bar-container';\n\n/** Event that is emitted when a snack bar is dismissed. */\nexport interface IBaoSnackBarDismiss {\n  /** Whether the snack bar was dismissed using the action button. */\n  dismissedByAction: boolean;\n}\n\n/** Maximum amount of milliseconds that can be passed into setTimeout. */\nconst MAX_TIMEOUT = Math.pow(2, 31) - 1;\n\n/**\n * Reference to a snack bar dispatched from the snack bar service.\n */\nexport class BaoSnackBarRef<T> {\n  /** The instance of the component making up the content of the snack bar. */\n  public instance: T;\n\n  /**\n   * The instance of the component making up the content of the snack bar.\n   * @docs-private\n   */\n  public containerInstance: IBaoSnackBarContainer;\n\n  /** Subject for notifying the user that the snack bar has been dismissed. */\n  private readonly _afterDismissed = new Subject<IBaoSnackBarDismiss>();\n\n  /** Subject for notifying the user that the snack bar has opened and appeared. */\n  private readonly _afterOpened = new Subject<void>();\n\n  /** Subject for notifying the user that the snack bar action was called. */\n  private readonly _onAction = new Subject<void>();\n\n  /**\n   * Timeout ID for the duration setTimeout call. Used to clear the timeout if the snackbar is\n   * dismissed before the duration passes.\n   */\n  private _durationTimeoutId: number;\n\n  /** Whether the snack bar was dismissed using the action button. */\n  private _dismissedByAction = false;\n\n  constructor(\n    containerInstance: IBaoSnackBarContainer,\n    private _overlayRef: OverlayRef\n  ) {\n    this.containerInstance = containerInstance;\n    // Dismiss snackbar on action.\n    this.onAction().subscribe(() => this.dismiss());\n    containerInstance._onExit.subscribe(() => this.finishDismiss());\n  }\n\n  /** Dismisses the snack bar. */\n  public dismiss(): void {\n    if (!this._afterDismissed.closed) {\n      this.containerInstance.exit();\n    }\n    clearTimeout(this._durationTimeoutId);\n  }\n\n  /** Marks the snackbar action clicked. */\n  public dismissWithAction(): void {\n    if (!this._onAction.closed) {\n      this._dismissedByAction = true;\n      this._onAction.next();\n      this._onAction.complete();\n    }\n  }\n\n  /** Dismisses the snack bar after some duration */\n  public dismissAfter(duration: number): void {\n    // Note that we need to cap the duration to the maximum value for setTimeout, because\n    // it'll revert to 1 if somebody passes in something greater (e.g. `Infinity`). See #17234.\n\n    // @TODO: window.setTimeout() ?\n    this._durationTimeoutId = window.setTimeout(\n      () => this.dismiss(),\n      Math.min(duration, MAX_TIMEOUT)\n    );\n  }\n\n  /** Marks the snackbar as opened */\n  public open(): void {\n    if (!this._afterOpened.closed) {\n      this._afterOpened.next();\n      this._afterOpened.complete();\n    }\n  }\n\n  /** Gets an observable that is notified when the snack bar is finished closing. */\n  public afterDismissed(): Observable<IBaoSnackBarDismiss> {\n    return this._afterDismissed;\n  }\n\n  /** Gets an observable that is notified when the snack bar has opened and appeared. */\n  public afterOpened(): Observable<void> {\n    return this.containerInstance._onEnter;\n  }\n\n  /** Gets an observable that is notified when the snack bar action is called. */\n  public onAction(): Observable<void> {\n    return this._onAction;\n  }\n\n  /** Cleans up the DOM after closing. */\n  private finishDismiss(): void {\n    this._overlayRef.dispose();\n\n    if (!this._onAction.closed) {\n      this._onAction.complete();\n    }\n\n    this._afterDismissed.next({ dismissedByAction: this._dismissedByAction });\n    this._afterDismissed.complete();\n    this._dismissedByAction = false;\n  }\n}\n"]}