UNPKG

@universal-material/angular

Version:

This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 16.1.0.

54 lines 11.9 kB
import { fromEvent, race } from 'rxjs'; import { delay, filter, map, takeUntil, tap, withLatestFrom } from 'rxjs/operators'; import { Key } from './key'; import { closest } from './util'; const isContainedIn = (element, array) => array ? array.some(item => item.contains(element)) : false; const matchesSelectorIfAny = (element, selector) => !selector || closest(element, selector) != null; // we have to add a more significant delay to avoid re-opening when handling (click) on a toggling element // TODO: use proper Angular platform detection when NgbAutoClose becomes a service and we can inject PLATFORM_ID const isMobile = (() => { const isIOS = () => /iPad|iPhone|iPod/.test(navigator.userAgent) || (/Macintosh/.test(navigator.userAgent) && navigator.maxTouchPoints && navigator.maxTouchPoints > 2); const isAndroid = () => /Android/.test(navigator.userAgent); return typeof navigator !== 'undefined' ? !!navigator.userAgent && (isIOS() || isAndroid()) : false; })(); // setting 'ngbAutoClose' synchronously on mobile results in immediate popup closing // when tapping on the triggering element const wrapAsyncForMobile = (fn) => isMobile ? () => setTimeout(() => fn(), 100) : fn; export function AutoClose(zone, document, type, close, closed$, insideElements, ignoreElements, insideSelector) { // closing on ESC and outside clicks if (type) { zone.runOutsideAngular(wrapAsyncForMobile(() => { const shouldCloseOnClick = (event) => { const element = event.target; if (event.button === 2 || isContainedIn(element, ignoreElements)) { return false; } if (type === 'inside') { return isContainedIn(element, insideElements) && matchesSelectorIfAny(element, insideSelector); } else if (type === 'outside') { return !isContainedIn(element, insideElements); } else /* if (type === true) */ { return matchesSelectorIfAny(element, insideSelector) || !isContainedIn(element, insideElements); } }; const escapes$ = fromEvent(document, 'keydown') .pipe(takeUntil(closed$), // tslint:disable-next-line:deprecation filter(e => e.which === Key.Escape), tap(e => e.preventDefault())); // we have to pre-calculate 'shouldCloseOnClick' on 'mousedown', // because on 'mouseup' DOM nodes might be detached const mouseDowns$ = fromEvent(document, 'mousedown').pipe(map(shouldCloseOnClick), takeUntil(closed$)); const closeableClicks$ = fromEvent(document, 'mouseup') .pipe(withLatestFrom(mouseDowns$), filter(([_, shouldClose]) => shouldClose), delay(0), takeUntil(closed$)); race([ escapes$.pipe(map(_ => 0 /* SOURCE.ESCAPE */)), closeableClicks$.pipe(map(_ => 1 /* SOURCE.CLICK */)) ]).subscribe({ next: source => zone.run(() => close(source)) }); })); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autoclose.js","sourceRoot":"","sources":["../../../../../src/util/autoclose.ts"],"names":[],"mappings":"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","sourcesContent":["import {NgZone} from '@angular/core';\r\nimport {fromEvent, Observable, race} from 'rxjs';\r\nimport {delay, filter, map, takeUntil, tap, withLatestFrom} from 'rxjs/operators';\r\nimport {Key} from './key';\r\nimport {closest} from './util';\r\n\r\nconst isContainedIn = (element: HTMLElement, array?: HTMLElement[]) =>\r\n  array ? array.some(item => item.contains(element)) : false;\r\n\r\nconst matchesSelectorIfAny = (element: HTMLElement, selector?: string) =>\r\n  !selector || closest(element, selector) != null;\r\n\r\n// we have to add a more significant delay to avoid re-opening when handling (click) on a toggling element\r\n// TODO: use proper Angular platform detection when NgbAutoClose becomes a service and we can inject PLATFORM_ID\r\nconst isMobile = (() => {\r\n  const isIOS = () => /iPad|iPhone|iPod/.test(navigator.userAgent) ||\r\n    (/Macintosh/.test(navigator.userAgent) && navigator.maxTouchPoints && navigator.maxTouchPoints > 2);\r\n  const isAndroid = () => /Android/.test(navigator.userAgent);\r\n\r\n  return typeof navigator !== 'undefined' ? !!navigator.userAgent && (isIOS() || isAndroid()) : false;\r\n})();\r\n\r\n// setting 'ngbAutoClose' synchronously on mobile results in immediate popup closing\r\n// when tapping on the triggering element\r\nconst wrapAsyncForMobile = (fn: any) => isMobile ? () => setTimeout(() => fn(), 100) : fn;\r\n\r\nexport const enum SOURCE {ESCAPE, CLICK}\r\n\r\nexport function AutoClose(\r\n  zone: NgZone, document: any, type: boolean | 'inside' | 'outside', close: (source: SOURCE) => void,\r\n  closed$: Observable<any>, insideElements: HTMLElement[], ignoreElements?: HTMLElement[], insideSelector?: string) {\r\n  // closing on ESC and outside clicks\r\n  if (type) {\r\n    zone.runOutsideAngular(wrapAsyncForMobile(() => {\r\n\r\n      const shouldCloseOnClick = (event: MouseEvent) => {\r\n        const element = event.target as HTMLElement;\r\n        if (event.button === 2 || isContainedIn(element, ignoreElements)) {\r\n          return false;\r\n        }\r\n        if (type === 'inside') {\r\n          return isContainedIn(element, insideElements) && matchesSelectorIfAny(element, insideSelector);\r\n        } else if (type === 'outside') {\r\n          return !isContainedIn(element, insideElements);\r\n        } else /* if (type === true) */ {\r\n          return matchesSelectorIfAny(element, insideSelector) || !isContainedIn(element, insideElements);\r\n        }\r\n      };\r\n\r\n      const escapes$ = fromEvent<KeyboardEvent>(document, 'keydown')\r\n        .pipe(\r\n          takeUntil(closed$),\r\n          // tslint:disable-next-line:deprecation\r\n          filter(e => e.which === Key.Escape), tap(e => e.preventDefault()));\r\n\r\n\r\n      // we have to pre-calculate 'shouldCloseOnClick' on 'mousedown',\r\n      // because on 'mouseup' DOM nodes might be detached\r\n      const mouseDowns$ =\r\n        fromEvent<MouseEvent>(document, 'mousedown').pipe(map(shouldCloseOnClick), takeUntil(closed$));\r\n\r\n      const closeableClicks$ = fromEvent<MouseEvent>(document, 'mouseup')\r\n        .pipe(\r\n          withLatestFrom(mouseDowns$), filter(([_, shouldClose]) => shouldClose), delay(0),\r\n          takeUntil(closed$)) as any as Observable<MouseEvent>;\r\n\r\n\r\n      race([\r\n        escapes$.pipe(map(_ => SOURCE.ESCAPE)), closeableClicks$.pipe(map(_ => SOURCE.CLICK))\r\n      ]).subscribe({\r\n        next: source => zone.run(() => close(source))\r\n      });\r\n    }));\r\n  }\r\n}\r\n"]}