modal-window.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { DOCUMENT } from '@angular/common';
  6. import { Component, ElementRef, EventEmitter, Inject, Input, NgZone, Output, ViewChild, ViewEncapsulation } from '@angular/core';
  7. import { fromEvent, Subject } from 'rxjs';
  8. import { filter, switchMap, take, takeUntil, tap } from 'rxjs/operators';
  9. import { getFocusableBoundaryElements } from '../util/focus-trap';
  10. import { Key } from '../util/key';
  11. import { ModalDismissReasons } from './modal-dismiss-reasons';
  12. var NgbModalWindow = /** @class */ (function () {
  13. function NgbModalWindow(_document, _elRef, _zone) {
  14. this._document = _document;
  15. this._elRef = _elRef;
  16. this._zone = _zone;
  17. this._closed$ = new Subject();
  18. this.backdrop = true;
  19. this.keyboard = true;
  20. this.dismissEvent = new EventEmitter();
  21. }
  22. /**
  23. * @param {?} reason
  24. * @return {?}
  25. */
  26. NgbModalWindow.prototype.dismiss = /**
  27. * @param {?} reason
  28. * @return {?}
  29. */
  30. function (reason) { this.dismissEvent.emit(reason); };
  31. /**
  32. * @return {?}
  33. */
  34. NgbModalWindow.prototype.ngOnInit = /**
  35. * @return {?}
  36. */
  37. function () { this._elWithFocus = this._document.activeElement; };
  38. /**
  39. * @return {?}
  40. */
  41. NgbModalWindow.prototype.ngAfterViewInit = /**
  42. * @return {?}
  43. */
  44. function () {
  45. var _this = this;
  46. var nativeElement = this._elRef.nativeElement;
  47. this._zone.runOutsideAngular((/**
  48. * @return {?}
  49. */
  50. function () {
  51. fromEvent(nativeElement, 'keydown')
  52. .pipe(takeUntil(_this._closed$),
  53. // tslint:disable-next-line:deprecation
  54. filter((/**
  55. * @param {?} e
  56. * @return {?}
  57. */
  58. function (e) { return e.which === Key.Escape && _this.keyboard; })))
  59. .subscribe((/**
  60. * @param {?} event
  61. * @return {?}
  62. */
  63. function (event) { return requestAnimationFrame((/**
  64. * @return {?}
  65. */
  66. function () {
  67. if (!event.defaultPrevented) {
  68. _this._zone.run((/**
  69. * @return {?}
  70. */
  71. function () { return _this.dismiss(ModalDismissReasons.ESC); }));
  72. }
  73. })); }));
  74. // We're listening to 'mousedown' and 'mouseup' to prevent modal from closing when pressing the mouse
  75. // inside the modal dialog and releasing it outside
  76. /** @type {?} */
  77. var preventClose = false;
  78. fromEvent(_this._dialogEl.nativeElement, 'mousedown')
  79. .pipe(takeUntil(_this._closed$), tap((/**
  80. * @return {?}
  81. */
  82. function () { return preventClose = false; })), switchMap((/**
  83. * @return {?}
  84. */
  85. function () { return fromEvent(nativeElement, 'mouseup').pipe(takeUntil(_this._closed$), take(1)); })), filter((/**
  86. * @param {?} __0
  87. * @return {?}
  88. */
  89. function (_a) {
  90. var target = _a.target;
  91. return nativeElement === target;
  92. })))
  93. .subscribe((/**
  94. * @return {?}
  95. */
  96. function () { preventClose = true; }));
  97. // We're listening to 'click' to dismiss modal on modal window click, except when:
  98. // 1. clicking on modal dialog itself
  99. // 2. closing was prevented by mousedown/up handlers
  100. // 3. clicking on scrollbar when the viewport is too small and modal doesn't fit (click is not triggered at all)
  101. fromEvent(nativeElement, 'click').pipe(takeUntil(_this._closed$)).subscribe((/**
  102. * @param {?} __0
  103. * @return {?}
  104. */
  105. function (_a) {
  106. var target = _a.target;
  107. if (_this.backdrop === true && nativeElement === target && !preventClose) {
  108. _this._zone.run((/**
  109. * @return {?}
  110. */
  111. function () { return _this.dismiss(ModalDismissReasons.BACKDROP_CLICK); }));
  112. }
  113. preventClose = false;
  114. }));
  115. }));
  116. if (!nativeElement.contains(document.activeElement)) {
  117. /** @type {?} */
  118. var autoFocusable = (/** @type {?} */ (nativeElement.querySelector("[ngbAutofocus]")));
  119. /** @type {?} */
  120. var firstFocusable = getFocusableBoundaryElements(nativeElement)[0];
  121. /** @type {?} */
  122. var elementToFocus = autoFocusable || firstFocusable || nativeElement;
  123. elementToFocus.focus();
  124. }
  125. };
  126. /**
  127. * @return {?}
  128. */
  129. NgbModalWindow.prototype.ngOnDestroy = /**
  130. * @return {?}
  131. */
  132. function () {
  133. var _this = this;
  134. /** @type {?} */
  135. var body = this._document.body;
  136. /** @type {?} */
  137. var elWithFocus = this._elWithFocus;
  138. /** @type {?} */
  139. var elementToFocus;
  140. if (elWithFocus && elWithFocus['focus'] && body.contains(elWithFocus)) {
  141. elementToFocus = elWithFocus;
  142. }
  143. else {
  144. elementToFocus = body;
  145. }
  146. this._zone.runOutsideAngular((/**
  147. * @return {?}
  148. */
  149. function () {
  150. setTimeout((/**
  151. * @return {?}
  152. */
  153. function () { return elementToFocus.focus(); }));
  154. _this._elWithFocus = null;
  155. }));
  156. this._closed$.next();
  157. };
  158. NgbModalWindow.decorators = [
  159. { type: Component, args: [{
  160. selector: 'ngb-modal-window',
  161. host: {
  162. '[class]': '"modal fade show d-block" + (windowClass ? " " + windowClass : "")',
  163. 'role': 'dialog',
  164. 'tabindex': '-1',
  165. '[attr.aria-modal]': 'true',
  166. '[attr.aria-labelledby]': 'ariaLabelledBy',
  167. },
  168. template: "\n <div #dialog [class]=\"'modal-dialog' + (size ? ' modal-' + size : '') + (centered ? ' modal-dialog-centered' : '') +\n (scrollable ? ' modal-dialog-scrollable' : '')\" role=\"document\">\n <div class=\"modal-content\"><ng-content></ng-content></div>\n </div>\n ",
  169. encapsulation: ViewEncapsulation.None,
  170. styles: ["ngb-modal-window .component-host-scrollable{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;overflow:hidden}"]
  171. }] }
  172. ];
  173. /** @nocollapse */
  174. NgbModalWindow.ctorParameters = function () { return [
  175. { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT,] }] },
  176. { type: ElementRef },
  177. { type: NgZone }
  178. ]; };
  179. NgbModalWindow.propDecorators = {
  180. _dialogEl: [{ type: ViewChild, args: ['dialog', { static: true },] }],
  181. ariaLabelledBy: [{ type: Input }],
  182. backdrop: [{ type: Input }],
  183. centered: [{ type: Input }],
  184. keyboard: [{ type: Input }],
  185. scrollable: [{ type: Input }],
  186. size: [{ type: Input }],
  187. windowClass: [{ type: Input }],
  188. dismissEvent: [{ type: Output, args: ['dismiss',] }]
  189. };
  190. return NgbModalWindow;
  191. }());
  192. export { NgbModalWindow };
  193. if (false) {
  194. /**
  195. * @type {?}
  196. * @private
  197. */
  198. NgbModalWindow.prototype._closed$;
  199. /**
  200. * @type {?}
  201. * @private
  202. */
  203. NgbModalWindow.prototype._elWithFocus;
  204. /**
  205. * @type {?}
  206. * @private
  207. */
  208. NgbModalWindow.prototype._dialogEl;
  209. /** @type {?} */
  210. NgbModalWindow.prototype.ariaLabelledBy;
  211. /** @type {?} */
  212. NgbModalWindow.prototype.backdrop;
  213. /** @type {?} */
  214. NgbModalWindow.prototype.centered;
  215. /** @type {?} */
  216. NgbModalWindow.prototype.keyboard;
  217. /** @type {?} */
  218. NgbModalWindow.prototype.scrollable;
  219. /** @type {?} */
  220. NgbModalWindow.prototype.size;
  221. /** @type {?} */
  222. NgbModalWindow.prototype.windowClass;
  223. /** @type {?} */
  224. NgbModalWindow.prototype.dismissEvent;
  225. /**
  226. * @type {?}
  227. * @private
  228. */
  229. NgbModalWindow.prototype._document;
  230. /**
  231. * @type {?}
  232. * @private
  233. */
  234. NgbModalWindow.prototype._elRef;
  235. /**
  236. * @type {?}
  237. * @private
  238. */
  239. NgbModalWindow.prototype._zone;
  240. }
  241. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-window.js","sourceRoot":"ng://@ng-bootstrap/ng-bootstrap/","sources":["modal/modal-window.ts"],"names":[],"mappings":"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","sourcesContent":["import {DOCUMENT} from '@angular/common';\nimport {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n  ViewChild,\n  ViewEncapsulation\n} from '@angular/core';\nimport {fromEvent, Subject} from 'rxjs';\nimport {filter, switchMap, take, takeUntil, tap} from 'rxjs/operators';\n\nimport {getFocusableBoundaryElements} from '../util/focus-trap';\nimport {Key} from '../util/key';\nimport {ModalDismissReasons} from './modal-dismiss-reasons';\n\n@Component({\n  selector: 'ngb-modal-window',\n  host: {\n    '[class]': '\"modal fade show d-block\" + (windowClass ? \" \" + windowClass : \"\")',\n    'role': 'dialog',\n    'tabindex': '-1',\n    '[attr.aria-modal]': 'true',\n    '[attr.aria-labelledby]': 'ariaLabelledBy',\n  },\n  template: `\n    <div #dialog [class]=\"'modal-dialog' + (size ? ' modal-' + size : '') + (centered ? ' modal-dialog-centered' : '') +\n     (scrollable ? ' modal-dialog-scrollable' : '')\" role=\"document\">\n        <div class=\"modal-content\"><ng-content></ng-content></div>\n    </div>\n    `,\n  encapsulation: ViewEncapsulation.None,\n  styleUrls: ['./modal.scss']\n})\nexport class NgbModalWindow implements OnInit,\n    AfterViewInit, OnDestroy {\n  private _closed$ = new Subject<void>();\n  private _elWithFocus: Element;  // element that is focused prior to modal opening\n\n  @ViewChild('dialog', {static: true}) private _dialogEl: ElementRef<HTMLElement>;\n\n  @Input() ariaLabelledBy: string;\n  @Input() backdrop: boolean | string = true;\n  @Input() centered: string;\n  @Input() keyboard = true;\n  @Input() scrollable: string;\n  @Input() size: string;\n  @Input() windowClass: string;\n\n  @Output('dismiss') dismissEvent = new EventEmitter();\n\n  constructor(\n      @Inject(DOCUMENT) private _document: any, private _elRef: ElementRef<HTMLElement>, private _zone: NgZone) {}\n\n  dismiss(reason): void { this.dismissEvent.emit(reason); }\n\n  ngOnInit() { this._elWithFocus = this._document.activeElement; }\n\n  ngAfterViewInit() {\n    const {nativeElement} = this._elRef;\n    this._zone.runOutsideAngular(() => {\n\n      fromEvent<KeyboardEvent>(nativeElement, 'keydown')\n          .pipe(\n              takeUntil(this._closed$),\n              // tslint:disable-next-line:deprecation\n              filter(e => e.which === Key.Escape && this.keyboard))\n          .subscribe(event => requestAnimationFrame(() => {\n                       if (!event.defaultPrevented) {\n                         this._zone.run(() => this.dismiss(ModalDismissReasons.ESC));\n                       }\n                     }));\n\n      // We're listening to 'mousedown' and 'mouseup' to prevent modal from closing when pressing the mouse\n      // inside the modal dialog and releasing it outside\n      let preventClose = false;\n      fromEvent<MouseEvent>(this._dialogEl.nativeElement, 'mousedown')\n          .pipe(\n              takeUntil(this._closed$), tap(() => preventClose = false),\n              switchMap(() => fromEvent<MouseEvent>(nativeElement, 'mouseup').pipe(takeUntil(this._closed$), take(1))),\n              filter(({target}) => nativeElement === target))\n          .subscribe(() => { preventClose = true; });\n\n      // We're listening to 'click' to dismiss modal on modal window click, except when:\n      // 1. clicking on modal dialog itself\n      // 2. closing was prevented by mousedown/up handlers\n      // 3. clicking on scrollbar when the viewport is too small and modal doesn't fit (click is not triggered at all)\n      fromEvent<MouseEvent>(nativeElement, 'click').pipe(takeUntil(this._closed$)).subscribe(({target}) => {\n        if (this.backdrop === true && nativeElement === target && !preventClose) {\n          this._zone.run(() => this.dismiss(ModalDismissReasons.BACKDROP_CLICK));\n        }\n        preventClose = false;\n      });\n    });\n\n    if (!nativeElement.contains(document.activeElement)) {\n      const autoFocusable = nativeElement.querySelector(`[ngbAutofocus]`) as HTMLElement;\n      const firstFocusable = getFocusableBoundaryElements(nativeElement)[0];\n\n      const elementToFocus = autoFocusable || firstFocusable || nativeElement;\n      elementToFocus.focus();\n    }\n  }\n\n  ngOnDestroy() {\n    const body = this._document.body;\n    const elWithFocus = this._elWithFocus;\n\n    let elementToFocus;\n    if (elWithFocus && elWithFocus['focus'] && body.contains(elWithFocus)) {\n      elementToFocus = elWithFocus;\n    } else {\n      elementToFocus = body;\n    }\n    this._zone.runOutsideAngular(() => {\n      setTimeout(() => elementToFocus.focus());\n      this._elWithFocus = null;\n    });\n\n    this._closed$.next();\n  }\n}\n"]}