UNPKG

@ngbracket/ngx-layout

Version:
192 lines 24.5 kB
/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import { DOCUMENT, isPlatformBrowser } from '@angular/common'; import { CSP_NONCE, Inject, Injectable, Optional, PLATFORM_ID, } from '@angular/core'; import { BehaviorSubject, merge, Observable } from 'rxjs'; import { filter } from 'rxjs/operators'; import { MediaChange } from '../media-change'; import * as i0 from "@angular/core"; /** * MediaMonitor configures listeners to mediaQuery changes and publishes an Observable facade to * convert mediaQuery change callbacks to subscriber notifications. These notifications will be * performed within the ng Zone to trigger change detections and component updates. * * NOTE: both mediaQuery activations and de-activations are announced in notifications */ export class MatchMedia { constructor(_zone, _platformId, _document, _nonce) { this._zone = _zone; this._platformId = _platformId; this._document = _document; this._nonce = _nonce; /** Initialize source with 'all' so all non-responsive APIs trigger style updates */ this.source = new BehaviorSubject(new MediaChange(true)); this.registry = new Map(); this.pendingRemoveListenerFns = []; this._observable$ = this.source.asObservable(); } /** * Publish list of all current activations */ get activations() { const results = []; this.registry.forEach((mql, key) => { if (mql.matches) { results.push(key); } }); return results; } /** * For the specified mediaQuery? */ isActive(mediaQuery) { const mql = this.registry.get(mediaQuery); return (mql?.matches ?? this.registerQuery(mediaQuery).some((m) => m.matches)); } /** * External observers can watch for all (or a specific) mql changes. * Typically used by the MediaQueryAdaptor; optionally available to components * who wish to use the MediaMonitor as mediaMonitor$ observable service. * * Use deferred registration process to register breakpoints only on subscription * This logic also enforces logic to register all mediaQueries BEFORE notify * subscribers of notifications. */ observe(mqList, filterOthers = false) { if (mqList && mqList.length) { const matchMedia$ = this._observable$.pipe(filter((change) => !filterOthers ? true : mqList.indexOf(change.mediaQuery) > -1)); const registration$ = new Observable((observer) => { // tslint:disable-line:max-line-length const matches = this.registerQuery(mqList); if (matches.length) { const lastChange = matches.pop(); matches.forEach((e) => { observer.next(e); }); this.source.next(lastChange); // last match is cached } observer.complete(); }); return merge(registration$, matchMedia$); } return this._observable$; } /** * Based on the BreakPointRegistry provider, register internal listeners for each unique * mediaQuery. Each listener emits specific MediaChange data to observers */ registerQuery(mediaQuery) { const list = Array.isArray(mediaQuery) ? mediaQuery : [mediaQuery]; const matches = []; buildQueryCss(list, this._document, this._nonce); list.forEach((query) => { const onMQLEvent = (e) => { this._zone.run(() => this.source.next(new MediaChange(e.matches, query))); }; let mql = this.registry.get(query); if (!mql) { mql = this.buildMQL(query); mql.addListener(onMQLEvent); this.pendingRemoveListenerFns.push(() => mql.removeListener(onMQLEvent)); this.registry.set(query, mql); } if (mql.matches) { matches.push(new MediaChange(true, query)); } }); return matches; } ngOnDestroy() { let fn; while ((fn = this.pendingRemoveListenerFns.pop())) { fn(); } } /** * Call window.matchMedia() to build a MediaQueryList; which * supports 0..n listeners for activation/deactivation */ buildMQL(query) { return constructMql(query, isPlatformBrowser(this._platformId)); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: MatchMedia, deps: [{ token: i0.NgZone }, { token: PLATFORM_ID }, { token: DOCUMENT }, { token: CSP_NONCE, optional: true }], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: MatchMedia, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: MatchMedia, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: () => [{ type: i0.NgZone }, { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID] }] }, { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [CSP_NONCE] }] }] }); /** * Private global registry for all dynamically-created, injected style tags * @see prepare(query) */ const ALL_STYLES = {}; /** * For Webkit engines that only trigger the MediaQueryList Listener * when there is at least one CSS selector for the respective media query. * * @param mediaQueries * @param _document */ function buildQueryCss(mediaQueries, _document, _nonce) { const list = mediaQueries.filter((it) => !ALL_STYLES[it]); if (list.length > 0) { const query = list.join(', '); try { const styleEl = _document.createElement('style'); styleEl.setAttribute('type', 'text/css'); if (_nonce) { styleEl.setAttribute('nonce', _nonce); } if (!styleEl.styleSheet) { const cssText = ` /* @ngbracket/ngx-layout - workaround for possible browser quirk with mediaQuery listeners see http://bit.ly/2sd4HMP */ @media ${query} {.fx-query-test{ }} `; styleEl.appendChild(_document.createTextNode(cssText)); } _document.head.appendChild(styleEl); // Store in private global registry list.forEach((mq) => (ALL_STYLES[mq] = styleEl)); } catch (e) { console.error(e); } } } function buildMockMql(query) { const et = new EventTarget(); et.matches = query === 'all' || query === ''; et.media = query; et.addListener = () => { }; et.removeListener = () => { }; et.addEventListener = () => { }; et.dispatchEvent = () => false; et.onchange = null; return et; } function constructMql(query, isBrowser) { const canListen = isBrowser && !!window.matchMedia('all').addListener; return canListen ? window.matchMedia(query) : buildMockMql(query); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"match-media.js","sourceRoot":"","sources":["../../../../../../projects/libs/flex-layout/core/match-media/match-media.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 */\nimport { DOCUMENT, isPlatformBrowser } from '@angular/common';\nimport {\n  CSP_NONCE,\n  Inject,\n  Injectable,\n  NgZone,\n  OnDestroy,\n  Optional,\n  PLATFORM_ID,\n} from '@angular/core';\nimport { BehaviorSubject, merge, Observable, Observer } from 'rxjs';\nimport { filter } from 'rxjs/operators';\n\nimport { MediaChange } from '../media-change';\n\n/**\n * MediaMonitor configures listeners to mediaQuery changes and publishes an Observable facade to\n * convert mediaQuery change callbacks to subscriber notifications. These notifications will be\n * performed within the ng Zone to trigger change detections and component updates.\n *\n * NOTE: both mediaQuery activations and de-activations are announced in notifications\n */\n@Injectable({ providedIn: 'root' })\nexport class MatchMedia implements OnDestroy {\n  /** Initialize source with 'all' so all non-responsive APIs trigger style updates */\n  readonly source = new BehaviorSubject<MediaChange>(new MediaChange(true));\n  registry = new Map<string, MediaQueryList>();\n  private readonly pendingRemoveListenerFns: Array<() => void> = [];\n\n  constructor(\n    protected _zone: NgZone,\n    @Inject(PLATFORM_ID) protected _platformId: Object,\n    @Inject(DOCUMENT) protected _document: any,\n    @Optional() @Inject(CSP_NONCE) protected _nonce?: string | null\n  ) {}\n\n  /**\n   * Publish list of all current activations\n   */\n  get activations(): string[] {\n    const results: string[] = [];\n    this.registry.forEach((mql: MediaQueryList, key: string) => {\n      if (mql.matches) {\n        results.push(key);\n      }\n    });\n    return results;\n  }\n\n  /**\n   * For the specified mediaQuery?\n   */\n  isActive(mediaQuery: string): boolean {\n    const mql = this.registry.get(mediaQuery);\n    return (\n      mql?.matches ?? this.registerQuery(mediaQuery).some((m) => m.matches)\n    );\n  }\n\n  /**\n   * External observers can watch for all (or a specific) mql changes.\n   *\n   * If a mediaQuery is not specified, then ALL mediaQuery activations will\n   * be announced.\n   */\n  observe(): Observable<MediaChange>;\n  observe(mediaQueries: string[]): Observable<MediaChange>;\n  observe(\n    mediaQueries: string[],\n    filterOthers: boolean\n  ): Observable<MediaChange>;\n\n  /**\n   * External observers can watch for all (or a specific) mql changes.\n   * Typically used by the MediaQueryAdaptor; optionally available to components\n   * who wish to use the MediaMonitor as mediaMonitor$ observable service.\n   *\n   * Use deferred registration process to register breakpoints only on subscription\n   * This logic also enforces logic to register all mediaQueries BEFORE notify\n   * subscribers of notifications.\n   */\n  observe(mqList?: string[], filterOthers = false): Observable<MediaChange> {\n    if (mqList && mqList.length) {\n      const matchMedia$: Observable<MediaChange> = this._observable$.pipe(\n        filter((change: MediaChange) =>\n          !filterOthers ? true : mqList.indexOf(change.mediaQuery) > -1\n        )\n      );\n      const registration$: Observable<MediaChange> = new Observable(\n        (observer: Observer<MediaChange>) => {\n          // tslint:disable-line:max-line-length\n          const matches: Array<MediaChange> = this.registerQuery(mqList);\n          if (matches.length) {\n            const lastChange = matches.pop()!;\n            matches.forEach((e: MediaChange) => {\n              observer.next(e);\n            });\n            this.source.next(lastChange); // last match is cached\n          }\n          observer.complete();\n        }\n      );\n      return merge(registration$, matchMedia$);\n    }\n\n    return this._observable$;\n  }\n\n  /**\n   * Based on the BreakPointRegistry provider, register internal listeners for each unique\n   * mediaQuery. Each listener emits specific MediaChange data to observers\n   */\n  registerQuery(mediaQuery: string | string[]) {\n    const list = Array.isArray(mediaQuery) ? mediaQuery : [mediaQuery];\n    const matches: MediaChange[] = [];\n\n    buildQueryCss(list, this._document, this._nonce);\n\n    list.forEach((query: string) => {\n      const onMQLEvent = (e: MediaQueryListEvent) => {\n        this._zone.run(() =>\n          this.source.next(new MediaChange(e.matches, query))\n        );\n      };\n\n      let mql = this.registry.get(query);\n      if (!mql) {\n        mql = this.buildMQL(query);\n        mql.addListener(onMQLEvent);\n        this.pendingRemoveListenerFns.push(() =>\n          mql!.removeListener(onMQLEvent)\n        );\n        this.registry.set(query, mql);\n      }\n\n      if (mql.matches) {\n        matches.push(new MediaChange(true, query));\n      }\n    });\n\n    return matches;\n  }\n\n  ngOnDestroy(): void {\n    let fn;\n    while ((fn = this.pendingRemoveListenerFns.pop())) {\n      fn();\n    }\n  }\n\n  /**\n   * Call window.matchMedia() to build a MediaQueryList; which\n   * supports 0..n listeners for activation/deactivation\n   */\n  protected buildMQL(query: string): MediaQueryList {\n    return constructMql(query, isPlatformBrowser(this._platformId));\n  }\n\n  protected _observable$ = this.source.asObservable();\n}\n\n/**\n * Private global registry for all dynamically-created, injected style tags\n * @see prepare(query)\n */\nconst ALL_STYLES: { [key: string]: any } = {};\n\n/**\n * For Webkit engines that only trigger the MediaQueryList Listener\n * when there is at least one CSS selector for the respective media query.\n *\n * @param mediaQueries\n * @param _document\n */\nfunction buildQueryCss(\n  mediaQueries: string[],\n  _document: Document,\n  _nonce?: string | null\n) {\n  const list = mediaQueries.filter((it) => !ALL_STYLES[it]);\n  if (list.length > 0) {\n    const query = list.join(', ');\n\n    try {\n      const styleEl = _document.createElement('style');\n\n      styleEl.setAttribute('type', 'text/css');\n      if (_nonce) {\n        styleEl.setAttribute('nonce', _nonce);\n      }\n      if (!(styleEl as any).styleSheet) {\n        const cssText = `\n/*\n  @ngbracket/ngx-layout - workaround for possible browser quirk with mediaQuery listeners\n  see http://bit.ly/2sd4HMP\n*/\n@media ${query} {.fx-query-test{ }}\n`;\n        styleEl.appendChild(_document.createTextNode(cssText));\n      }\n\n      _document.head!.appendChild(styleEl);\n\n      // Store in private global registry\n      list.forEach((mq) => (ALL_STYLES[mq] = styleEl));\n    } catch (e) {\n      console.error(e);\n    }\n  }\n}\n\nfunction buildMockMql(query: string) {\n  const et: any = new EventTarget();\n  et.matches = query === 'all' || query === '';\n  et.media = query;\n  et.addListener = () => {};\n  et.removeListener = () => {};\n  et.addEventListener = () => {};\n  et.dispatchEvent = () => false;\n  et.onchange = null;\n\n  return et as MediaQueryList;\n}\n\nfunction constructMql(query: string, isBrowser: boolean): MediaQueryList {\n  const canListen =\n    isBrowser && !!(<Window>window).matchMedia('all').addListener;\n\n  return canListen ? (<Window>window).matchMedia(query) : buildMockMql(query);\n}\n"]}