[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 { Directive, Input, Optional, ElementRef, } from '@angular/core';
|
---|
| 9 | import { MatDialog } from './dialog';
|
---|
| 10 | import { _closeDialogVia, MatDialogRef } from './dialog-ref';
|
---|
| 11 | /** Counter used to generate unique IDs for dialog elements. */
|
---|
| 12 | let dialogElementUid = 0;
|
---|
| 13 | /**
|
---|
| 14 | * Button that will close the current dialog.
|
---|
| 15 | */
|
---|
| 16 | export class MatDialogClose {
|
---|
| 17 | constructor(
|
---|
| 18 | /**
|
---|
| 19 | * Reference to the containing dialog.
|
---|
| 20 | * @deprecated `dialogRef` property to become private.
|
---|
| 21 | * @breaking-change 13.0.0
|
---|
| 22 | */
|
---|
| 23 | // The dialog title directive is always used in combination with a `MatDialogRef`.
|
---|
| 24 | // tslint:disable-next-line: lightweight-tokens
|
---|
| 25 | dialogRef, _elementRef, _dialog) {
|
---|
| 26 | this.dialogRef = dialogRef;
|
---|
| 27 | this._elementRef = _elementRef;
|
---|
| 28 | this._dialog = _dialog;
|
---|
| 29 | /** Default to "button" to prevents accidental form submits. */
|
---|
| 30 | this.type = 'button';
|
---|
| 31 | }
|
---|
| 32 | ngOnInit() {
|
---|
| 33 | if (!this.dialogRef) {
|
---|
| 34 | // When this directive is included in a dialog via TemplateRef (rather than being
|
---|
| 35 | // in a Component), the DialogRef isn't available via injection because embedded
|
---|
| 36 | // views cannot be given a custom injector. Instead, we look up the DialogRef by
|
---|
| 37 | // ID. This must occur in `onInit`, as the ID binding for the dialog container won't
|
---|
| 38 | // be resolved at constructor time.
|
---|
| 39 | this.dialogRef = getClosestDialog(this._elementRef, this._dialog.openDialogs);
|
---|
| 40 | }
|
---|
| 41 | }
|
---|
| 42 | ngOnChanges(changes) {
|
---|
| 43 | const proxiedChange = changes['_matDialogClose'] || changes['_matDialogCloseResult'];
|
---|
| 44 | if (proxiedChange) {
|
---|
| 45 | this.dialogResult = proxiedChange.currentValue;
|
---|
| 46 | }
|
---|
| 47 | }
|
---|
| 48 | _onButtonClick(event) {
|
---|
| 49 | // Determinate the focus origin using the click event, because using the FocusMonitor will
|
---|
| 50 | // result in incorrect origins. Most of the time, close buttons will be auto focused in the
|
---|
| 51 | // dialog, and therefore clicking the button won't result in a focus change. This means that
|
---|
| 52 | // the FocusMonitor won't detect any origin change, and will always output `program`.
|
---|
| 53 | _closeDialogVia(this.dialogRef, event.screenX === 0 && event.screenY === 0 ? 'keyboard' : 'mouse', this.dialogResult);
|
---|
| 54 | }
|
---|
| 55 | }
|
---|
| 56 | MatDialogClose.decorators = [
|
---|
| 57 | { type: Directive, args: [{
|
---|
| 58 | selector: '[mat-dialog-close], [matDialogClose]',
|
---|
| 59 | exportAs: 'matDialogClose',
|
---|
| 60 | host: {
|
---|
| 61 | '(click)': '_onButtonClick($event)',
|
---|
| 62 | '[attr.aria-label]': 'ariaLabel || null',
|
---|
| 63 | '[attr.type]': 'type',
|
---|
| 64 | }
|
---|
| 65 | },] }
|
---|
| 66 | ];
|
---|
| 67 | MatDialogClose.ctorParameters = () => [
|
---|
| 68 | { type: MatDialogRef, decorators: [{ type: Optional }] },
|
---|
| 69 | { type: ElementRef },
|
---|
| 70 | { type: MatDialog }
|
---|
| 71 | ];
|
---|
| 72 | MatDialogClose.propDecorators = {
|
---|
| 73 | ariaLabel: [{ type: Input, args: ['aria-label',] }],
|
---|
| 74 | type: [{ type: Input }],
|
---|
| 75 | dialogResult: [{ type: Input, args: ['mat-dialog-close',] }],
|
---|
| 76 | _matDialogClose: [{ type: Input, args: ['matDialogClose',] }]
|
---|
| 77 | };
|
---|
| 78 | /**
|
---|
| 79 | * Title of a dialog element. Stays fixed to the top of the dialog when scrolling.
|
---|
| 80 | */
|
---|
| 81 | export class MatDialogTitle {
|
---|
| 82 | constructor(
|
---|
| 83 | // The dialog title directive is always used in combination with a `MatDialogRef`.
|
---|
| 84 | // tslint:disable-next-line: lightweight-tokens
|
---|
| 85 | _dialogRef, _elementRef, _dialog) {
|
---|
| 86 | this._dialogRef = _dialogRef;
|
---|
| 87 | this._elementRef = _elementRef;
|
---|
| 88 | this._dialog = _dialog;
|
---|
| 89 | /** Unique id for the dialog title. If none is supplied, it will be auto-generated. */
|
---|
| 90 | this.id = `mat-dialog-title-${dialogElementUid++}`;
|
---|
| 91 | }
|
---|
| 92 | ngOnInit() {
|
---|
| 93 | if (!this._dialogRef) {
|
---|
| 94 | this._dialogRef = getClosestDialog(this._elementRef, this._dialog.openDialogs);
|
---|
| 95 | }
|
---|
| 96 | if (this._dialogRef) {
|
---|
| 97 | Promise.resolve().then(() => {
|
---|
| 98 | const container = this._dialogRef._containerInstance;
|
---|
| 99 | if (container && !container._ariaLabelledBy) {
|
---|
| 100 | container._ariaLabelledBy = this.id;
|
---|
| 101 | }
|
---|
| 102 | });
|
---|
| 103 | }
|
---|
| 104 | }
|
---|
| 105 | }
|
---|
| 106 | MatDialogTitle.decorators = [
|
---|
| 107 | { type: Directive, args: [{
|
---|
| 108 | selector: '[mat-dialog-title], [matDialogTitle]',
|
---|
| 109 | exportAs: 'matDialogTitle',
|
---|
| 110 | host: {
|
---|
| 111 | 'class': 'mat-dialog-title',
|
---|
| 112 | '[id]': 'id',
|
---|
| 113 | },
|
---|
| 114 | },] }
|
---|
| 115 | ];
|
---|
| 116 | MatDialogTitle.ctorParameters = () => [
|
---|
| 117 | { type: MatDialogRef, decorators: [{ type: Optional }] },
|
---|
| 118 | { type: ElementRef },
|
---|
| 119 | { type: MatDialog }
|
---|
| 120 | ];
|
---|
| 121 | MatDialogTitle.propDecorators = {
|
---|
| 122 | id: [{ type: Input }]
|
---|
| 123 | };
|
---|
| 124 | /**
|
---|
| 125 | * Scrollable content container of a dialog.
|
---|
| 126 | */
|
---|
| 127 | export class MatDialogContent {
|
---|
| 128 | }
|
---|
| 129 | MatDialogContent.decorators = [
|
---|
| 130 | { type: Directive, args: [{
|
---|
| 131 | selector: `[mat-dialog-content], mat-dialog-content, [matDialogContent]`,
|
---|
| 132 | host: { 'class': 'mat-dialog-content' }
|
---|
| 133 | },] }
|
---|
| 134 | ];
|
---|
| 135 | /**
|
---|
| 136 | * Container for the bottom action buttons in a dialog.
|
---|
| 137 | * Stays fixed to the bottom when scrolling.
|
---|
| 138 | */
|
---|
| 139 | export class MatDialogActions {
|
---|
| 140 | }
|
---|
| 141 | MatDialogActions.decorators = [
|
---|
| 142 | { type: Directive, args: [{
|
---|
| 143 | selector: `[mat-dialog-actions], mat-dialog-actions, [matDialogActions]`,
|
---|
| 144 | host: { 'class': 'mat-dialog-actions' }
|
---|
| 145 | },] }
|
---|
| 146 | ];
|
---|
| 147 | /**
|
---|
| 148 | * Finds the closest MatDialogRef to an element by looking at the DOM.
|
---|
| 149 | * @param element Element relative to which to look for a dialog.
|
---|
| 150 | * @param openDialogs References to the currently-open dialogs.
|
---|
| 151 | */
|
---|
| 152 | function getClosestDialog(element, openDialogs) {
|
---|
| 153 | let parent = element.nativeElement.parentElement;
|
---|
| 154 | while (parent && !parent.classList.contains('mat-dialog-container')) {
|
---|
| 155 | parent = parent.parentElement;
|
---|
| 156 | }
|
---|
| 157 | return parent ? openDialogs.find(dialog => dialog.id === parent.id) : null;
|
---|
| 158 | }
|
---|
| 159 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog-content-directives.js","sourceRoot":"","sources":["../../../../../../src/material/dialog/dialog-content-directives.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  Directive,\n  Input,\n  OnChanges,\n  OnInit,\n  Optional,\n  SimpleChanges,\n  ElementRef,\n} from '@angular/core';\nimport {MatDialog} from './dialog';\nimport {_closeDialogVia, MatDialogRef} from './dialog-ref';\n\n/** Counter used to generate unique IDs for dialog elements. */\nlet dialogElementUid = 0;\n\n/**\n * Button that will close the current dialog.\n */\n@Directive({\n  selector: '[mat-dialog-close], [matDialogClose]',\n  exportAs: 'matDialogClose',\n  host: {\n    '(click)': '_onButtonClick($event)',\n    '[attr.aria-label]': 'ariaLabel || null',\n    '[attr.type]': 'type',\n  }\n})\nexport class MatDialogClose implements OnInit, OnChanges {\n  /** Screenreader label for the button. */\n  @Input('aria-label') ariaLabel: string;\n\n  /** Default to \"button\" to prevents accidental form submits. */\n  @Input() type: 'submit' | 'button' | 'reset' = 'button';\n\n  /** Dialog close input. */\n  @Input('mat-dialog-close') dialogResult: any;\n\n  @Input('matDialogClose') _matDialogClose: any;\n\n  constructor(\n\n    /**\n     * Reference to the containing dialog.\n     * @deprecated `dialogRef` property to become private.\n     * @breaking-change 13.0.0\n     */\n    // The dialog title directive is always used in combination with a `MatDialogRef`.\n    // tslint:disable-next-line: lightweight-tokens\n    @Optional() public dialogRef: MatDialogRef<any>,\n    private _elementRef: ElementRef<HTMLElement>,\n    private _dialog: MatDialog) {}\n\n  ngOnInit() {\n    if (!this.dialogRef) {\n      // When this directive is included in a dialog via TemplateRef (rather than being\n      // in a Component), the DialogRef isn't available via injection because embedded\n      // views cannot be given a custom injector. Instead, we look up the DialogRef by\n      // ID. This must occur in `onInit`, as the ID binding for the dialog container won't\n      // be resolved at constructor time.\n      this.dialogRef = getClosestDialog(this._elementRef, this._dialog.openDialogs)!;\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    const proxiedChange = changes['_matDialogClose'] || changes['_matDialogCloseResult'];\n\n    if (proxiedChange) {\n      this.dialogResult = proxiedChange.currentValue;\n    }\n  }\n\n  _onButtonClick(event: MouseEvent) {\n    // Determinate the focus origin using the click event, because using the FocusMonitor will\n    // result in incorrect origins. Most of the time, close buttons will be auto focused in the\n    // dialog, and therefore clicking the button won't result in a focus change. This means that\n    // the FocusMonitor won't detect any origin change, and will always output `program`.\n    _closeDialogVia(this.dialogRef,\n        event.screenX === 0 && event.screenY === 0 ? 'keyboard' : 'mouse', this.dialogResult);\n  }\n}\n\n/**\n * Title of a dialog element. Stays fixed to the top of the dialog when scrolling.\n */\n@Directive({\n  selector: '[mat-dialog-title], [matDialogTitle]',\n  exportAs: 'matDialogTitle',\n  host: {\n    'class': 'mat-dialog-title',\n    '[id]': 'id',\n  },\n})\nexport class MatDialogTitle implements OnInit {\n  /** Unique id for the dialog title. If none is supplied, it will be auto-generated. */\n  @Input() id: string = `mat-dialog-title-${dialogElementUid++}`;\n\n  constructor(\n      // The dialog title directive is always used in combination with a `MatDialogRef`.\n      // tslint:disable-next-line: lightweight-tokens\n      @Optional() private _dialogRef: MatDialogRef<any>,\n      private _elementRef: ElementRef<HTMLElement>,\n      private _dialog: MatDialog) {}\n\n  ngOnInit() {\n    if (!this._dialogRef) {\n      this._dialogRef = getClosestDialog(this._elementRef, this._dialog.openDialogs)!;\n    }\n\n    if (this._dialogRef) {\n      Promise.resolve().then(() => {\n        const container = this._dialogRef._containerInstance;\n\n        if (container && !container._ariaLabelledBy) {\n          container._ariaLabelledBy = this.id;\n        }\n      });\n    }\n  }\n}\n\n\n/**\n * Scrollable content container of a dialog.\n */\n@Directive({\n  selector: `[mat-dialog-content], mat-dialog-content, [matDialogContent]`,\n  host: {'class': 'mat-dialog-content'}\n})\nexport class MatDialogContent {}\n\n\n/**\n * Container for the bottom action buttons in a dialog.\n * Stays fixed to the bottom when scrolling.\n */\n@Directive({\n  selector: `[mat-dialog-actions], mat-dialog-actions, [matDialogActions]`,\n  host: {'class': 'mat-dialog-actions'}\n})\nexport class MatDialogActions {}\n\n\n/**\n * Finds the closest MatDialogRef to an element by looking at the DOM.\n * @param element Element relative to which to look for a dialog.\n * @param openDialogs References to the currently-open dialogs.\n */\nfunction getClosestDialog(element: ElementRef<HTMLElement>, openDialogs: MatDialogRef<any>[]) {\n  let parent: HTMLElement | null = element.nativeElement.parentElement;\n\n  while (parent && !parent.classList.contains('mat-dialog-container')) {\n    parent = parent.parentElement;\n  }\n\n  return parent ? openDialogs.find(dialog => dialog.id === parent!.id) : null;\n}\n"]} |
---|