UNPKG

@ngbracket/ngx-layout

Version:
265 lines 35.8 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 { Inject, Injectable } from '@angular/core'; import { mergeAlias } from '../add-alias'; import { MediaChange } from '../media-change'; import { LAYOUT_CONFIG } from '../tokens/library-config'; import { sortDescendingPriority } from '../utils/sort'; import { DOCUMENT } from '@angular/common'; import * as i0 from "@angular/core"; import * as i1 from "../breakpoints/break-point-registry"; const PRINT = 'print'; export const BREAKPOINT_PRINT = { alias: PRINT, mediaQuery: PRINT, priority: 1000 }; /** * PrintHook - Use to intercept print MediaQuery activations and force * layouts to render with the specified print alias/breakpoint * * Used in MediaMarshaller and MediaObserver */ export class PrintHook { constructor(breakpoints, layoutConfig, _document) { this.breakpoints = breakpoints; this.layoutConfig = layoutConfig; this._document = _document; // registeredBeforeAfterPrintHooks tracks if we registered the `beforeprint` // and `afterprint` event listeners. this.registeredBeforeAfterPrintHooks = false; // isPrintingBeforeAfterEvent is used to track if we are printing from within // a `beforeprint` event handler. This prevents the typical `stopPrinting` // form `interceptEvents` so that printing is not stopped while the dialog // is still open. This is an extension of the `isPrinting` property on // browsers which support `beforeprint` and `afterprint` events. this.isPrintingBeforeAfterEvent = false; this.beforePrintEventListeners = []; this.afterPrintEventListeners = []; this.formerActivations = null; // Is this service currently in print mode this.isPrinting = false; this.queue = new PrintQueue(); this.deactivations = []; } /** Add 'print' mediaQuery: to listen for matchMedia activations */ withPrintQuery(queries) { return [...queries, PRINT]; } /** Is the MediaChange event for any 'print' @media */ isPrintEvent(e) { return e.mediaQuery.startsWith(PRINT); } /** What is the desired mqAlias to use while printing? */ get printAlias() { return [...(this.layoutConfig.printWithBreakpoints ?? [])]; } /** Lookup breakpoints associated with print aliases. */ get printBreakPoints() { return this.printAlias .map(alias => this.breakpoints.findByAlias(alias)) .filter(bp => bp !== null); } /** Lookup breakpoint associated with mediaQuery */ getEventBreakpoints({ mediaQuery }) { const bp = this.breakpoints.findByQuery(mediaQuery); const list = bp ? [...this.printBreakPoints, bp] : this.printBreakPoints; return list.sort(sortDescendingPriority); } /** Update event with printAlias mediaQuery information */ updateEvent(event) { let bp = this.breakpoints.findByQuery(event.mediaQuery); if (this.isPrintEvent(event)) { // Reset from 'print' to first (highest priority) print breakpoint bp = this.getEventBreakpoints(event)[0]; event.mediaQuery = bp?.mediaQuery ?? ''; } return mergeAlias(event, bp); } // registerBeforeAfterPrintHooks registers a `beforeprint` event hook so we can // trigger print styles synchronously and apply proper layout styles. // It is a noop if the hooks have already been registered or if the document's // `defaultView` is not available. registerBeforeAfterPrintHooks(target) { // `defaultView` may be null when rendering on the server or in other contexts. if (!this._document.defaultView || this.registeredBeforeAfterPrintHooks) { return; } this.registeredBeforeAfterPrintHooks = true; const beforePrintListener = () => { // If we aren't already printing, start printing and update the styles as // if there was a regular print `MediaChange`(from matchMedia). if (!this.isPrinting) { this.isPrintingBeforeAfterEvent = true; this.startPrinting(target, this.getEventBreakpoints(new MediaChange(true, PRINT))); target.updateStyles(); } }; const afterPrintListener = () => { // If we aren't already printing, start printing and update the styles as // if there was a regular print `MediaChange`(from matchMedia). this.isPrintingBeforeAfterEvent = false; if (this.isPrinting) { this.stopPrinting(target); target.updateStyles(); } }; // Could we have teardown logic to remove if there are no print listeners being used? this._document.defaultView.addEventListener('beforeprint', beforePrintListener); this._document.defaultView.addEventListener('afterprint', afterPrintListener); this.beforePrintEventListeners.push(beforePrintListener); this.afterPrintEventListeners.push(afterPrintListener); } /** * Prepare RxJS tap operator with partial application * @return pipeable tap predicate */ interceptEvents(target) { return (event) => { if (this.isPrintEvent(event)) { if (event.matches && !this.isPrinting) { this.startPrinting(target, this.getEventBreakpoints(event)); target.updateStyles(); } else if (!event.matches && this.isPrinting && !this.isPrintingBeforeAfterEvent) { this.stopPrinting(target); target.updateStyles(); } return; } this.collectActivations(target, event); }; } /** Stop mediaChange event propagation in event streams */ blockPropagation() { return (event) => { return !(this.isPrinting || this.isPrintEvent(event)); }; } /** * Save current activateBreakpoints (for later restore) * and substitute only the printAlias breakpoint */ startPrinting(target, bpList) { this.isPrinting = true; this.formerActivations = target.activatedBreakpoints; target.activatedBreakpoints = this.queue.addPrintBreakpoints(bpList); } /** For any print de-activations, reset the entire print queue */ stopPrinting(target) { target.activatedBreakpoints = this.deactivations; this.deactivations = []; this.formerActivations = null; this.queue.clear(); this.isPrinting = false; } /** * To restore pre-Print Activations, we must capture the proper * list of breakpoint activations BEFORE print starts. OnBeforePrint() * is supported; so 'print' mediaQuery activations are used as a fallback * in browsers without `beforeprint` support. * * > But activated breakpoints are deactivated BEFORE 'print' activation. * * Let's capture all de-activations using the following logic: * * When not printing: * - clear cache when activating non-print breakpoint * - update cache (and sort) when deactivating * * When printing: * - sort and save when starting print * - restore as activatedTargets and clear when stop printing */ collectActivations(target, event) { if (!this.isPrinting || this.isPrintingBeforeAfterEvent) { if (!this.isPrintingBeforeAfterEvent) { // Only clear deactivations if we aren't printing from a `beforeprint` event. // Otherwise, this will clear before `stopPrinting()` is called to restore // the pre-Print Activations. this.deactivations = []; return; } if (!event.matches) { const bp = this.breakpoints.findByQuery(event.mediaQuery); // Deactivating a breakpoint if (bp) { const hasFormerBp = this.formerActivations && this.formerActivations.includes(bp); const wasActivated = !this.formerActivations && target.activatedBreakpoints.includes(bp); const shouldDeactivate = hasFormerBp || wasActivated; if (shouldDeactivate) { this.deactivations.push(bp); this.deactivations.sort(sortDescendingPriority); } } } } } /** Teardown logic for the service. */ ngOnDestroy() { if (this._document.defaultView) { this.beforePrintEventListeners.forEach(l => this._document.defaultView.removeEventListener('beforeprint', l)); this.afterPrintEventListeners.forEach(l => this._document.defaultView.removeEventListener('afterprint', l)); } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: PrintHook, deps: [{ token: i1.BreakPointRegistry }, { token: LAYOUT_CONFIG }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: PrintHook, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: PrintHook, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: () => [{ type: i1.BreakPointRegistry }, { type: undefined, decorators: [{ type: Inject, args: [LAYOUT_CONFIG] }] }, { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT] }] }] }); // ************************************************************************ // Internal Utility class 'PrintQueue' // ************************************************************************ /** * Utility class to manage print breakpoints + activatedBreakpoints * with correct sorting WHILE printing */ class PrintQueue { constructor() { /** Sorted queue with prioritized print breakpoints */ this.printBreakpoints = []; } addPrintBreakpoints(bpList) { bpList.push(BREAKPOINT_PRINT); bpList.sort(sortDescendingPriority); bpList.forEach(bp => this.addBreakpoint(bp)); return this.printBreakpoints; } /** Add Print breakpoint to queue */ addBreakpoint(bp) { if (!!bp) { const bpInList = this.printBreakpoints.find(it => it.mediaQuery === bp.mediaQuery); if (bpInList === undefined) { // If this is a `printAlias` breakpoint, then append. If a true 'print' breakpoint, // register as highest priority in the queue this.printBreakpoints = isPrintBreakPoint(bp) ? [bp, ...this.printBreakpoints] : [...this.printBreakpoints, bp]; } } } /** Restore original activated breakpoints and clear internal caches */ clear() { this.printBreakpoints = []; } } // ************************************************************************ // Internal Utility methods // ************************************************************************ /** Only support intercept queueing if the Breakpoint is a print @media query */ function isPrintBreakPoint(bp) { return bp?.mediaQuery.startsWith(PRINT) ?? false; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"print-hook.js","sourceRoot":"","sources":["../../../../../../projects/libs/flex-layout/core/media-marshaller/print-hook.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 {Inject, Injectable, OnDestroy} from '@angular/core';\n\nimport {mergeAlias} from '../add-alias';\nimport {MediaChange} from '../media-change';\nimport {BreakPoint} from '../breakpoints/break-point';\nimport {LAYOUT_CONFIG, LayoutConfigOptions} from '../tokens/library-config';\nimport {BreakPointRegistry, OptionalBreakPoint} from '../breakpoints/break-point-registry';\nimport {sortDescendingPriority} from '../utils/sort';\nimport {DOCUMENT} from '@angular/common';\n\n/**\n * Interface to apply PrintHook to call anonymous `target.updateStyles()`\n */\nexport interface HookTarget {\n  activatedBreakpoints: BreakPoint[];\n  updateStyles(): void;\n}\n\nconst PRINT = 'print';\nexport const BREAKPOINT_PRINT = {\n  alias: PRINT,\n  mediaQuery: PRINT,\n  priority: 1000\n};\n\n/**\n * PrintHook - Use to intercept print MediaQuery activations and force\n *             layouts to render with the specified print alias/breakpoint\n *\n * Used in MediaMarshaller and MediaObserver\n */\n@Injectable({providedIn: 'root'})\nexport class PrintHook implements OnDestroy {\n  constructor(\n      protected breakpoints: BreakPointRegistry,\n      @Inject(LAYOUT_CONFIG) protected layoutConfig: LayoutConfigOptions,\n      @Inject(DOCUMENT) protected _document: any) {\n  }\n\n  /** Add 'print' mediaQuery: to listen for matchMedia activations */\n  withPrintQuery(queries: string[]): string[] {\n    return [...queries, PRINT];\n  }\n\n  /** Is the MediaChange event for any 'print' @media */\n  isPrintEvent(e: MediaChange): boolean {\n    return e.mediaQuery.startsWith(PRINT);\n  }\n\n  /** What is the desired mqAlias to use while printing? */\n  get printAlias(): string[] {\n    return [...(this.layoutConfig.printWithBreakpoints ?? [])];\n  }\n\n  /** Lookup breakpoints associated with print aliases. */\n  get printBreakPoints(): BreakPoint[] {\n    return this.printAlias\n        .map(alias => this.breakpoints.findByAlias(alias))\n        .filter(bp => bp !== null) as BreakPoint[];\n  }\n\n  /** Lookup breakpoint associated with mediaQuery */\n  getEventBreakpoints({mediaQuery}: MediaChange): BreakPoint[] {\n    const bp = this.breakpoints.findByQuery(mediaQuery);\n    const list = bp ? [...this.printBreakPoints, bp] : this.printBreakPoints;\n\n    return list.sort(sortDescendingPriority);\n  }\n\n  /** Update event with printAlias mediaQuery information */\n  updateEvent(event: MediaChange): MediaChange {\n    let bp: OptionalBreakPoint = this.breakpoints.findByQuery(event.mediaQuery);\n\n    if (this.isPrintEvent(event)) {\n      // Reset from 'print' to first (highest priority) print breakpoint\n      bp = this.getEventBreakpoints(event)[0];\n      event.mediaQuery = bp?.mediaQuery ?? '';\n    }\n\n    return mergeAlias(event, bp);\n  }\n\n\n  // registeredBeforeAfterPrintHooks tracks if we registered the `beforeprint`\n  //  and `afterprint` event listeners.\n  private registeredBeforeAfterPrintHooks = false;\n\n  // isPrintingBeforeAfterEvent is used to track if we are printing from within\n  // a `beforeprint` event handler. This prevents the typical `stopPrinting`\n  // form `interceptEvents` so that printing is not stopped while the dialog\n  // is still open. This is an extension of the `isPrinting` property on\n  // browsers which support `beforeprint` and `afterprint` events.\n  private isPrintingBeforeAfterEvent = false;\n\n  private beforePrintEventListeners: Function[] = [];\n  private afterPrintEventListeners: Function[] = [];\n\n  private formerActivations: Array<BreakPoint> | null = null;\n\n  // registerBeforeAfterPrintHooks registers a `beforeprint` event hook so we can\n  // trigger print styles synchronously and apply proper layout styles.\n  // It is a noop if the hooks have already been registered or if the document's\n  // `defaultView` is not available.\n  registerBeforeAfterPrintHooks(target: HookTarget) {\n    // `defaultView` may be null when rendering on the server or in other contexts.\n    if (!this._document.defaultView || this.registeredBeforeAfterPrintHooks) {\n      return;\n    }\n\n    this.registeredBeforeAfterPrintHooks = true;\n\n    const beforePrintListener = () => {\n      // If we aren't already printing, start printing and update the styles as\n      // if there was a regular print `MediaChange`(from matchMedia).\n      if (!this.isPrinting) {\n        this.isPrintingBeforeAfterEvent = true;\n        this.startPrinting(target, this.getEventBreakpoints(new MediaChange(true, PRINT)));\n        target.updateStyles();\n      }\n    };\n\n    const afterPrintListener = () => {\n      // If we aren't already printing, start printing and update the styles as\n      // if there was a regular print `MediaChange`(from matchMedia).\n      this.isPrintingBeforeAfterEvent = false;\n      if (this.isPrinting) {\n        this.stopPrinting(target);\n        target.updateStyles();\n      }\n    };\n\n    // Could we have teardown logic to remove if there are no print listeners being used?\n    this._document.defaultView.addEventListener('beforeprint', beforePrintListener);\n    this._document.defaultView.addEventListener('afterprint', afterPrintListener);\n\n    this.beforePrintEventListeners.push(beforePrintListener);\n    this.afterPrintEventListeners.push(afterPrintListener);\n  }\n\n  /**\n   * Prepare RxJS tap operator with partial application\n   * @return pipeable tap predicate\n   */\n  interceptEvents(target: HookTarget) {\n    return (event: MediaChange) => {\n      if (this.isPrintEvent(event)) {\n        if (event.matches && !this.isPrinting) {\n          this.startPrinting(target, this.getEventBreakpoints(event));\n          target.updateStyles();\n        } else if (!event.matches && this.isPrinting && !this.isPrintingBeforeAfterEvent) {\n          this.stopPrinting(target);\n          target.updateStyles();\n        }\n\n        return;\n      }\n\n      this.collectActivations(target, event);\n    };\n  }\n\n  /** Stop mediaChange event propagation in event streams */\n  blockPropagation() {\n    return (event: MediaChange): boolean => {\n      return !(this.isPrinting || this.isPrintEvent(event));\n    };\n  }\n\n  /**\n   * Save current activateBreakpoints (for later restore)\n   * and substitute only the printAlias breakpoint\n   */\n  protected startPrinting(target: HookTarget, bpList: OptionalBreakPoint[]) {\n    this.isPrinting = true;\n    this.formerActivations = target.activatedBreakpoints;\n    target.activatedBreakpoints = this.queue.addPrintBreakpoints(bpList);\n  }\n\n  /** For any print de-activations, reset the entire print queue */\n  protected stopPrinting(target: HookTarget) {\n    target.activatedBreakpoints = this.deactivations;\n    this.deactivations = [];\n    this.formerActivations = null;\n    this.queue.clear();\n    this.isPrinting = false;\n  }\n\n  /**\n   * To restore pre-Print Activations, we must capture the proper\n   * list of breakpoint activations BEFORE print starts. OnBeforePrint()\n   * is supported; so 'print' mediaQuery activations are used as a fallback\n   * in browsers without `beforeprint` support.\n   *\n   * >  But activated breakpoints are deactivated BEFORE 'print' activation.\n   *\n   * Let's capture all de-activations using the following logic:\n   *\n   *  When not printing:\n   *    - clear cache when activating non-print breakpoint\n   *    - update cache (and sort) when deactivating\n   *\n   *  When printing:\n   *    - sort and save when starting print\n   *    - restore as activatedTargets and clear when stop printing\n   */\n  collectActivations(target: HookTarget, event: MediaChange) {\n    if (!this.isPrinting || this.isPrintingBeforeAfterEvent) {\n      if (!this.isPrintingBeforeAfterEvent) {\n        // Only clear deactivations if we aren't printing from a `beforeprint` event.\n        // Otherwise, this will clear before `stopPrinting()` is called to restore\n        // the pre-Print Activations.\n        this.deactivations = [];\n\n        return;\n      }\n\n      if (!event.matches) {\n        const bp = this.breakpoints.findByQuery(event.mediaQuery);\n        // Deactivating a breakpoint\n        if (bp) {\n          const hasFormerBp = this.formerActivations && this.formerActivations.includes(bp);\n          const wasActivated = !this.formerActivations && target.activatedBreakpoints.includes(bp);\n          const shouldDeactivate = hasFormerBp || wasActivated;\n          if (shouldDeactivate) {\n            this.deactivations.push(bp);\n            this.deactivations.sort(sortDescendingPriority);\n          }\n        }\n      }\n    }\n  }\n\n  /** Teardown logic for the service. */\n  ngOnDestroy() {\n    if (this._document.defaultView) {\n      this.beforePrintEventListeners.forEach(l => this._document.defaultView.removeEventListener('beforeprint', l));\n      this.afterPrintEventListeners.forEach(l => this._document.defaultView.removeEventListener('afterprint', l));\n    }\n  }\n\n  // Is this service currently in print mode\n  private isPrinting = false;\n  private queue = new PrintQueue();\n  private deactivations: BreakPoint[] = [];\n}\n\n// ************************************************************************\n// Internal Utility class 'PrintQueue'\n// ************************************************************************\n\n/**\n * Utility class to manage print breakpoints + activatedBreakpoints\n * with correct sorting WHILE printing\n */\nclass PrintQueue {\n  /** Sorted queue with prioritized print breakpoints */\n  printBreakpoints: BreakPoint[] = [];\n\n  addPrintBreakpoints(bpList: OptionalBreakPoint[]): BreakPoint[] {\n    bpList.push(BREAKPOINT_PRINT);\n    bpList.sort(sortDescendingPriority);\n    bpList.forEach(bp => this.addBreakpoint(bp));\n\n    return this.printBreakpoints;\n  }\n\n  /** Add Print breakpoint to queue */\n  addBreakpoint(bp: OptionalBreakPoint) {\n    if (!!bp) {\n      const bpInList = this.printBreakpoints.find(it => it.mediaQuery === bp.mediaQuery);\n\n      if (bpInList === undefined) {\n        // If this is a `printAlias` breakpoint, then append. If a true 'print' breakpoint,\n        // register as highest priority in the queue\n        this.printBreakpoints = isPrintBreakPoint(bp) ? [bp, ...this.printBreakpoints]\n            : [...this.printBreakpoints, bp];\n      }\n    }\n  }\n\n  /** Restore original activated breakpoints and clear internal caches */\n  clear() {\n    this.printBreakpoints = [];\n  }\n}\n\n// ************************************************************************\n// Internal Utility methods\n// ************************************************************************\n\n/** Only support intercept queueing if the Breakpoint is a print @media query */\nfunction isPrintBreakPoint(bp: OptionalBreakPoint): boolean {\n  return bp?.mediaQuery.startsWith(PRINT) ?? false;\n}\n"]}