UNPKG

ngx-extended-pdf-viewer

Version:

Embedding PDF files in your Angular application. Highly configurable viewer including the toolbar, sidebar, and all the features you're used to.

179 lines 30.2 kB
import { isPlatformBrowser } from '@angular/common'; import { Component, EventEmitter, HostListener, Inject, Input, Output, PLATFORM_ID, effect, } from '@angular/core'; import * as i0 from "@angular/core"; import * as i1 from "./../../pdf-notification-service"; import * as i2 from "../../toolbar/pdf-shy-button/pdf-shy-button-service"; import * as i3 from "../../ngx-extended-pdf-viewer.service"; import * as i4 from "@angular/common"; import * as i5 from "../../responsive-visibility"; export class PdfSecondaryToolbarComponent { element; notificationService; platformId; pdfShyButtonService; ngxExtendedPdfViewerService; customSecondaryToolbar; secondaryToolbarTop; mobileFriendlyZoomScale; localizationInitialized; spreadChange = new EventEmitter(); disablePreviousPage = true; disableNextPage = true; classMutationObserver; PDFViewerApplication; constructor(element, notificationService, platformId, pdfShyButtonService, ngxExtendedPdfViewerService) { this.element = element; this.notificationService = notificationService; this.platformId = platformId; this.pdfShyButtonService = pdfShyButtonService; this.ngxExtendedPdfViewerService = ngxExtendedPdfViewerService; effect(() => { this.PDFViewerApplication = notificationService.onPDFJSInitSignal(); if (this.PDFViewerApplication) { this.onPdfJsInit(); } }); } onPdfJsInit() { this.PDFViewerApplication?.eventBus.on('pagechanging', () => { this.updateUIState(); }); this.PDFViewerApplication?.eventBus.on('pagerendered', () => { this.updateUIState(); }); } updateUIState() { setTimeout(() => { const currentPage = this.PDFViewerApplication?.pdfViewer.currentPageNumber; const previousButton = document.getElementById('previousPage'); if (previousButton) { this.disablePreviousPage = Number(currentPage) <= 1; previousButton.disabled = this.disablePreviousPage; } const nextButton = document.getElementById('nextPage'); if (nextButton) { this.disableNextPage = currentPage === this.PDFViewerApplication?.pagesCount; nextButton.disabled = this.disableNextPage; } }); } onSpreadChange(newSpread) { this.spreadChange.emit(newSpread); } ngOnChanges(changes) { setTimeout(() => this.checkVisibility()); } onResize() { setTimeout(() => this.checkVisibility()); } ngAfterViewInit() { if (isPlatformBrowser(this.platformId)) { const targetNode = this.element.nativeElement; const config = { attributes: true, childList: true, subtree: true }; this.classMutationObserver = new MutationObserver((mutationList, observer) => { for (const mutation of mutationList) { if (mutation.type === 'attributes') { if (mutation.attributeName === 'class') { this.checkVisibility(); break; } } else if (mutation.type === 'childList') { this.checkVisibility(); break; } } }); this.classMutationObserver.observe(targetNode, config); } } ngOnDestroy() { if (this.classMutationObserver) { this.classMutationObserver.disconnect(); this.classMutationObserver = undefined; } } checkVisibility() { let visibleButtons = 0; const e = this.element.nativeElement; const f = e.children.item(0); if (f) { const g = f.children.item(0); if (g && g instanceof HTMLElement) { visibleButtons = this.checkVisibilityRecursively(g); } } this.ngxExtendedPdfViewerService.secondaryMenuIsEmpty = visibleButtons === 0; } checkVisibilityRecursively(e) { if (typeof window === 'undefined') { // server-side rendering return 0; } if (e.style.display === 'none') { return 0; } if (e.classList.contains('hidden')) { return 0; } if (e.classList.contains('invisible')) { return 0; } const style = window.getComputedStyle(e); if (style.display === 'none') { return 0; } if (e instanceof HTMLButtonElement || e instanceof HTMLAnchorElement) { return 1; } let count = 0; const children = e.children; if (children?.length) { for (let i = 0; i < children.length && count === 0; i++) { const child = children.item(i); if (child && child instanceof HTMLElement) { count += this.checkVisibilityRecursively(child); } } } return count; } onClick(htmlevent, action, eventBusName, closeOnClick) { const origin = htmlevent.target; origin?.classList.add('toggled'); if (action) { action.call(this, htmlevent, true); htmlevent.preventDefault(); } else if (eventBusName) { this.PDFViewerApplication?.eventBus.dispatch(eventBusName); htmlevent.preventDefault(); } if (closeOnClick) { this.PDFViewerApplication?.secondaryToolbar.close(); } } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PdfSecondaryToolbarComponent, deps: [{ token: i0.ElementRef }, { token: i1.PDFNotificationService }, { token: PLATFORM_ID }, { token: i2.PdfShyButtonService }, { token: i3.NgxExtendedPdfViewerService }], target: i0.ɵɵFactoryTarget.Component }); static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: PdfSecondaryToolbarComponent, selector: "pdf-secondary-toolbar", inputs: { customSecondaryToolbar: "customSecondaryToolbar", secondaryToolbarTop: "secondaryToolbarTop", mobileFriendlyZoomScale: "mobileFriendlyZoomScale", localizationInitialized: "localizationInitialized" }, outputs: { spreadChange: "spreadChange" }, host: { listeners: { "window:resize": "onResize()" } }, usesOnChanges: true, ngImport: i0, template: "<ng-container [ngTemplateOutlet]=\"customSecondaryToolbar ? customSecondaryToolbar : defaultSecondaryToolbar\">\n</ng-container>\n\n<ng-template #defaultSecondaryToolbar>\n <div id=\"secondaryToolbar\" class=\"secondaryToolbar hidden doorHangerRight\" [style.top]=\"secondaryToolbarTop\"\n [style.transform]=\"'scale(' + mobileFriendlyZoomScale + ')'\" [style.transformOrigin]=\"'right top'\">\n <div id=\"secondaryToolbarButtonContainer\">\n @for (button of pdfShyButtonService.buttons; track button.id) {\n <button type=\"button\" [id]=\"button.id\" [ngClass]=\"button.cssClass | invertForSecondaryToolbar\"\n [class.toggled]=\"button.toggled\" class=\"secondaryToolbarButton\" [title]=\"button.title\"\n [attr.data-l10n-id]=\"button.l10nId\"\n (click)=\"onClick($event, button.action, button.eventBusName, button.closeOnClick)\"\n [attr.aria-label]=\"button.title\" [disabled]=\"button.disabled\">\n @if (button.image) {\n <span class=\"icon\" role=\"img\" aria-hidden=\"true\" [attr.aria-label]=\"button.title\"\n [innerHTML]=\"button.image\"></span>\n }\n @else {\n <span class=\"icon\" role=\"img\" aria-hidden=\"true\" [attr.aria-label]=\"button.title\"></span>\n }\n <span class=\"toolbar-caption\" [attr.data-l10n-id]=\"button.l10nLabel\">{{ button.title }}</span>\n </button>\n }\n </div>\n </div>\n</ng-template>", styles: ["svg{position:absolute;display:inline-block;top:0;left:0}.secondaryToolbarButton{display:inline-flex;align-items:center;justify-content:flex-start;border:0 none;background:none;width:calc(100% - 4px);height:25px;position:relative;margin:0 0 4px;padding:3px 0 1px;min-height:25px;white-space:normal}.secondaryToolbarButton span{display:inline-block}.secondaryToolbarButton[disabled]{opacity:.5}::ng-deep html[dir=ltr] ngx-extended-pdf-viewer .secondaryToolbarButton{padding-left:4px;text-align:left}::ng-deep html[dir=rtl] ngx-extended-pdf-viewer .secondaryToolbarButton{padding-right:4px;text-align:right}::ng-deep html[dir=ltr] ngx-extended-pdf-viewer .secondaryToolbarButton>span{padding-right:4px}::ng-deep html[dir=rtl] ngx-extended-pdf-viewer .secondaryToolbarButton>span{padding-left:4px}.secondaryToolbar{height:auto;z-index:30000}::ng-deep html[dir=ltr] ngx-extended-pdf-viewer .secondaryToolbar{right:4px}::ng-deep [dir=rtl] ngx-extended-pdf-viewer .secondaryToolbar{left:4px}#secondaryToolbarButtonContainer{width:250px;max-height:775px;overflow-y:auto;-webkit-overflow-scrolling:touch}.toolbar-caption{position:relative;top:-3px}.icon{width:24px}\n"], dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i5.NegativeResponsiveCSSClassPipe, name: "invertForSecondaryToolbar" }] }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PdfSecondaryToolbarComponent, decorators: [{ type: Component, args: [{ selector: 'pdf-secondary-toolbar', template: "<ng-container [ngTemplateOutlet]=\"customSecondaryToolbar ? customSecondaryToolbar : defaultSecondaryToolbar\">\n</ng-container>\n\n<ng-template #defaultSecondaryToolbar>\n <div id=\"secondaryToolbar\" class=\"secondaryToolbar hidden doorHangerRight\" [style.top]=\"secondaryToolbarTop\"\n [style.transform]=\"'scale(' + mobileFriendlyZoomScale + ')'\" [style.transformOrigin]=\"'right top'\">\n <div id=\"secondaryToolbarButtonContainer\">\n @for (button of pdfShyButtonService.buttons; track button.id) {\n <button type=\"button\" [id]=\"button.id\" [ngClass]=\"button.cssClass | invertForSecondaryToolbar\"\n [class.toggled]=\"button.toggled\" class=\"secondaryToolbarButton\" [title]=\"button.title\"\n [attr.data-l10n-id]=\"button.l10nId\"\n (click)=\"onClick($event, button.action, button.eventBusName, button.closeOnClick)\"\n [attr.aria-label]=\"button.title\" [disabled]=\"button.disabled\">\n @if (button.image) {\n <span class=\"icon\" role=\"img\" aria-hidden=\"true\" [attr.aria-label]=\"button.title\"\n [innerHTML]=\"button.image\"></span>\n }\n @else {\n <span class=\"icon\" role=\"img\" aria-hidden=\"true\" [attr.aria-label]=\"button.title\"></span>\n }\n <span class=\"toolbar-caption\" [attr.data-l10n-id]=\"button.l10nLabel\">{{ button.title }}</span>\n </button>\n }\n </div>\n </div>\n</ng-template>", styles: ["svg{position:absolute;display:inline-block;top:0;left:0}.secondaryToolbarButton{display:inline-flex;align-items:center;justify-content:flex-start;border:0 none;background:none;width:calc(100% - 4px);height:25px;position:relative;margin:0 0 4px;padding:3px 0 1px;min-height:25px;white-space:normal}.secondaryToolbarButton span{display:inline-block}.secondaryToolbarButton[disabled]{opacity:.5}::ng-deep html[dir=ltr] ngx-extended-pdf-viewer .secondaryToolbarButton{padding-left:4px;text-align:left}::ng-deep html[dir=rtl] ngx-extended-pdf-viewer .secondaryToolbarButton{padding-right:4px;text-align:right}::ng-deep html[dir=ltr] ngx-extended-pdf-viewer .secondaryToolbarButton>span{padding-right:4px}::ng-deep html[dir=rtl] ngx-extended-pdf-viewer .secondaryToolbarButton>span{padding-left:4px}.secondaryToolbar{height:auto;z-index:30000}::ng-deep html[dir=ltr] ngx-extended-pdf-viewer .secondaryToolbar{right:4px}::ng-deep [dir=rtl] ngx-extended-pdf-viewer .secondaryToolbar{left:4px}#secondaryToolbarButtonContainer{width:250px;max-height:775px;overflow-y:auto;-webkit-overflow-scrolling:touch}.toolbar-caption{position:relative;top:-3px}.icon{width:24px}\n"] }] }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i1.PDFNotificationService }, { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID] }] }, { type: i2.PdfShyButtonService }, { type: i3.NgxExtendedPdfViewerService }], propDecorators: { customSecondaryToolbar: [{ type: Input }], secondaryToolbarTop: [{ type: Input }], mobileFriendlyZoomScale: [{ type: Input }], localizationInitialized: [{ type: Input }], spreadChange: [{ type: Output }], onResize: [{ type: HostListener, args: ['window:resize'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pdf-secondary-toolbar.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-extended-pdf-viewer/src/lib/secondary-toolbar/pdf-secondary-toolbar/pdf-secondary-toolbar.component.ts","../../../../../../projects/ngx-extended-pdf-viewer/src/lib/secondary-toolbar/pdf-secondary-toolbar/pdf-secondary-toolbar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Output,\n  PLATFORM_ID,\n  SimpleChanges,\n  TemplateRef,\n  effect,\n} from '@angular/core';\nimport { NgxExtendedPdfViewerService } from '../../ngx-extended-pdf-viewer.service';\nimport { IPDFViewerApplication } from '../../options/pdf-viewer-application';\nimport { PdfShyButtonService } from '../../toolbar/pdf-shy-button/pdf-shy-button-service';\nimport { PDFNotificationService } from './../../pdf-notification-service';\n\n@Component({\n  selector: 'pdf-secondary-toolbar',\n  templateUrl: './pdf-secondary-toolbar.component.html',\n  styleUrls: ['./pdf-secondary-toolbar.component.css'],\n})\nexport class PdfSecondaryToolbarComponent implements OnChanges, AfterViewInit, OnDestroy {\n  @Input()\n  public customSecondaryToolbar: TemplateRef<any> | undefined;\n\n  @Input()\n  public secondaryToolbarTop;\n\n  @Input()\n  public mobileFriendlyZoomScale: number;\n\n  @Input()\n  public localizationInitialized: boolean;\n\n  @Output()\n  public spreadChange = new EventEmitter<'off' | 'even' | 'odd'>();\n\n  public disablePreviousPage = true;\n\n  public disableNextPage = true;\n\n  private classMutationObserver: MutationObserver | undefined;\n\n  private PDFViewerApplication: IPDFViewerApplication | undefined;\n\n  constructor(\n    private element: ElementRef,\n    public notificationService: PDFNotificationService,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    public pdfShyButtonService: PdfShyButtonService,\n    private ngxExtendedPdfViewerService: NgxExtendedPdfViewerService\n  ) {\n    effect(() => {\n      this.PDFViewerApplication = notificationService.onPDFJSInitSignal();\n      if (this.PDFViewerApplication) {\n        this.onPdfJsInit();\n      }\n    });\n  }\n\n  public onPdfJsInit(): void {\n    this.PDFViewerApplication?.eventBus.on('pagechanging', () => {\n      this.updateUIState();\n    });\n    this.PDFViewerApplication?.eventBus.on('pagerendered', () => {\n      this.updateUIState();\n    });\n  }\n\n  public updateUIState(): void {\n    setTimeout(() => {\n      const currentPage = this.PDFViewerApplication?.pdfViewer.currentPageNumber;\n      const previousButton = document.getElementById('previousPage') as HTMLButtonElement;\n      if (previousButton) {\n        this.disablePreviousPage = Number(currentPage) <= 1;\n        previousButton.disabled = this.disablePreviousPage;\n      }\n      const nextButton = document.getElementById('nextPage') as HTMLButtonElement;\n      if (nextButton) {\n        this.disableNextPage = currentPage === this.PDFViewerApplication?.pagesCount;\n        nextButton.disabled = this.disableNextPage;\n      }\n    });\n  }\n\n  public onSpreadChange(newSpread: 'off' | 'odd' | 'even'): void {\n    this.spreadChange.emit(newSpread);\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    setTimeout(() => this.checkVisibility());\n  }\n\n  @HostListener('window:resize')\n  public onResize() {\n    setTimeout(() => this.checkVisibility());\n  }\n\n  public ngAfterViewInit() {\n    if (isPlatformBrowser(this.platformId)) {\n      const targetNode = this.element.nativeElement as HTMLElement;\n\n      const config = { attributes: true, childList: true, subtree: true };\n\n      this.classMutationObserver = new MutationObserver((mutationList: MutationRecord[], observer) => {\n        for (const mutation of mutationList) {\n          if (mutation.type === 'attributes') {\n            if (mutation.attributeName === 'class') {\n              this.checkVisibility();\n              break;\n            }\n          } else if (mutation.type === 'childList') {\n            this.checkVisibility();\n            break;\n          }\n        }\n      });\n\n      this.classMutationObserver.observe(targetNode, config);\n    }\n  }\n\n  public ngOnDestroy(): void {\n    if (this.classMutationObserver) {\n      this.classMutationObserver.disconnect();\n      this.classMutationObserver = undefined;\n    }\n  }\n\n  public checkVisibility(): void {\n    let visibleButtons = 0;\n    const e = this.element.nativeElement as HTMLElement;\n    const f = e.children.item(0);\n    if (f) {\n      const g = f.children.item(0);\n      if (g && g instanceof HTMLElement) {\n        visibleButtons = this.checkVisibilityRecursively(g);\n      }\n    }\n    this.ngxExtendedPdfViewerService.secondaryMenuIsEmpty = visibleButtons === 0;\n  }\n\n  private checkVisibilityRecursively(e: HTMLElement): number {\n    if (typeof window === 'undefined') {\n      // server-side rendering\n      return 0;\n    }\n    if (e.style.display === 'none') {\n      return 0;\n    }\n    if (e.classList.contains('hidden')) {\n      return 0;\n    }\n    if (e.classList.contains('invisible')) {\n      return 0;\n    }\n\n    const style = window.getComputedStyle(e);\n    if (style.display === 'none') {\n      return 0;\n    }\n\n    if (e instanceof HTMLButtonElement || e instanceof HTMLAnchorElement) {\n      return 1;\n    }\n    let count = 0;\n    const children = e.children;\n    if (children?.length) {\n      for (let i = 0; i < children.length && count === 0; i++) {\n        const child = children.item(i);\n        if (child && child instanceof HTMLElement) {\n          count += this.checkVisibilityRecursively(child);\n        }\n      }\n    }\n    return count;\n  }\n\n  public onClick(\n    htmlevent: Event,\n    action: undefined | ((htmlevent: Event, secondaryToolbar: boolean) => void),\n    eventBusName?: string,\n    closeOnClick?: boolean\n  ): void {\n    const origin = htmlevent.target as HTMLElement;\n    origin?.classList.add('toggled');\n    if (action) {\n      action.call(this, htmlevent, true);\n      htmlevent.preventDefault();\n    } else if (eventBusName) {\n      this.PDFViewerApplication?.eventBus.dispatch(eventBusName);\n      htmlevent.preventDefault();\n    }\n    if (closeOnClick) {\n      this.PDFViewerApplication?.secondaryToolbar.close();\n    }\n  }\n}\n","<ng-container [ngTemplateOutlet]=\"customSecondaryToolbar ? customSecondaryToolbar : defaultSecondaryToolbar\">\n</ng-container>\n\n<ng-template #defaultSecondaryToolbar>\n  <div id=\"secondaryToolbar\" class=\"secondaryToolbar hidden doorHangerRight\" [style.top]=\"secondaryToolbarTop\"\n    [style.transform]=\"'scale(' + mobileFriendlyZoomScale + ')'\" [style.transformOrigin]=\"'right top'\">\n    <div id=\"secondaryToolbarButtonContainer\">\n      @for (button of pdfShyButtonService.buttons; track button.id) {\n      <button type=\"button\" [id]=\"button.id\" [ngClass]=\"button.cssClass | invertForSecondaryToolbar\"\n        [class.toggled]=\"button.toggled\" class=\"secondaryToolbarButton\" [title]=\"button.title\"\n        [attr.data-l10n-id]=\"button.l10nId\"\n        (click)=\"onClick($event, button.action, button.eventBusName, button.closeOnClick)\"\n        [attr.aria-label]=\"button.title\" [disabled]=\"button.disabled\">\n        @if (button.image) {\n        <span class=\"icon\" role=\"img\" aria-hidden=\"true\" [attr.aria-label]=\"button.title\"\n          [innerHTML]=\"button.image\"></span>\n        }\n        @else {\n        <span class=\"icon\" role=\"img\" aria-hidden=\"true\" [attr.aria-label]=\"button.title\"></span>\n        }\n        <span class=\"toolbar-caption\" [attr.data-l10n-id]=\"button.l10nLabel\">{{ button.title }}</span>\n      </button>\n      }\n    </div>\n  </div>\n</ng-template>"]}