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.

224 lines 25.4 kB
import { Component, ContentChild, Input, ViewChild, effect } from '@angular/core'; import * as i0 from "@angular/core"; import * as i1 from "./pdf-shy-button-service"; import * as i2 from "@angular/platform-browser"; import * as i3 from "../../pdf-notification-service"; import * as i4 from "../../pdf-csp-policy.service"; export class PdfShyButtonComponent { pdfShyButtonServiceService; sanitizer; renderer; pdfCspPolicyService; primaryToolbarId; secondaryMenuId; cssClass = 'invisible'; eventBusName = undefined; l10nId; l10nLabel; title; toggled; disabled; order; action = undefined; closeOnClick = true; onlySecondaryMenu = false; ariaHasPopup = false; ariaControls; role; PDFViewerApplication; renderContent = false; buttonRef; nestedContent = null; _imageHtml; get imageHtml() { if (this._imageHtml) { // allow non-literal svg tags (sanitized in the setter) return this.sanitizer.bypassSecurityTrustHtml(this._imageHtml); // NOSONAR } return undefined; } set image(value) { const svgTags = [ // 'a' is not allowed! 'animate', 'animateMotion', 'animateTransform', 'audio', 'canvas', 'circle', 'clipPath', 'defs', 'desc', 'discard', 'ellipse', 'feBlend', 'feColorMatrix', 'feComponentTransfer', 'feComposite', 'feConvolveMatrix', 'feDiffuseLighting', 'feDisplacementMap', 'feDistantLight', 'feDropShadow', 'feFlood', 'feFuncA', 'feFuncB', 'feFuncG', 'feFuncR', 'feGaussianBlur', 'feImage', 'feMerge', 'feMergeNode', 'feMorphology', 'feOffset', 'fePointLight', 'feSpecularLighting', 'feSpotLight', 'feTile', 'feTurbulence', 'filter', 'foreignObject', 'g', 'iframe', 'image', 'line', 'linearGradient', 'marker', 'mask', 'metadata', 'mpath', 'path', 'pattern', 'polygon', 'polyline', 'radialGradient', 'rect', 'script', 'set', 'stop', 'style', 'svg', 'switch', 'symbol', 'text', 'textPath', 'title', 'tspan', 'unknown', 'use', 'video', 'view', ]; // only <svg> and SVG tags are allowed const tags = value.split('<').filter((tag) => tag.length > 0); const legal = tags.every((tag) => tag.startsWith('svg') || tag.startsWith('/') || svgTags.includes(tag.split(/\s|>/)[0])); if (!legal) { throw new Error('Illegal image for PDFShyButton. Only SVG images are allowed. Please use only the tags <svg> and <path>. ' + value); } this._imageHtml = this.pdfCspPolicyService.sanitizeHTML(value); } constructor(pdfShyButtonServiceService, sanitizer, renderer, notificationService, pdfCspPolicyService) { this.pdfShyButtonServiceService = pdfShyButtonServiceService; this.sanitizer = sanitizer; this.renderer = renderer; this.pdfCspPolicyService = pdfCspPolicyService; effect(() => { this.PDFViewerApplication = notificationService.onPDFJSInitSignal(); }); } ngAfterViewInit() { this.updateButtonImage(); } ngOnInit() { this.pdfShyButtonServiceService.add(this); } ngOnChanges(changes) { this.pdfShyButtonServiceService.update(this); } onClick(htmlEvent) { if (htmlEvent instanceof KeyboardEvent && htmlEvent.key !== 'Enter' && htmlEvent.key !== ' ') { return; } if (this.action) { this.action(htmlEvent, false); htmlEvent.preventDefault(); } else if (this.eventBusName) { this.PDFViewerApplication?.eventBus.dispatch(this.eventBusName); htmlEvent.preventDefault(); } } updateButtonImage() { if (this.buttonRef) { const el = this.buttonRef.nativeElement; if (this._imageHtml) { const temp = this.renderer.createElement('div'); this.pdfCspPolicyService.addTrustedHTML(temp, this._imageHtml); const image = temp.children[0]; if (!el.innerHTML.includes(image.innerHTML)) { // if using SSR, the HTML code may already be there this.renderer.appendChild(el, image); } } else { const childNodes = el.childNodes; for (let child of childNodes) { this.renderer.removeChild(el, child); } } } } ngAfterContentInit() { if (this.primaryToolbarId === 'nestedComponent') { this.renderContent = !!this.nestedContent; console.log('renderContent', this.renderContent); } } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PdfShyButtonComponent, deps: [{ token: i1.PdfShyButtonService }, { token: i2.DomSanitizer }, { token: i0.Renderer2 }, { token: i3.PDFNotificationService }, { token: i4.PdfCspPolicyService }], target: i0.ɵɵFactoryTarget.Component }); static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: PdfShyButtonComponent, selector: "pdf-shy-button", inputs: { primaryToolbarId: "primaryToolbarId", secondaryMenuId: "secondaryMenuId", cssClass: "cssClass", eventBusName: "eventBusName", l10nId: "l10nId", l10nLabel: "l10nLabel", title: "title", toggled: "toggled", disabled: "disabled", order: "order", action: "action", closeOnClick: "closeOnClick", onlySecondaryMenu: "onlySecondaryMenu", ariaHasPopup: "ariaHasPopup", ariaControls: "ariaControls", role: "role", image: "image" }, queries: [{ propertyName: "nestedContent", first: true, predicate: ["nestedContent"], descendants: true }], viewQueries: [{ propertyName: "buttonRef", first: true, predicate: ["buttonRef"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "@if (!onlySecondaryMenu) {\n@if (nestedContent) {\n<span (click)=\"onClick($event)\" onKeyDown=\"onClick($event)\" [class]=\"cssClass\">\n <ng-content></ng-content>\n</span>\n}\n@else {\n<button type=\"button\" [id]=\"primaryToolbarId\" class=\"toolbarButton\" [class]=\"cssClass\" [title]=\"title\"\n [attr.data-l10n-id]=\"l10nId\" [class.toggled]=\"toggled\" [disabled]=\"disabled\" (click)=\"onClick($event)\"\n [attr.aria-label]=\"title\" [attr.role]=\"role\" [attr.aria-expanded]=\"toggled ? 'true' : (ariaHasPopup ? 'false' : null)\"\n [attr.aria-haspopup]=\"ariaHasPopup\" [attr.aria-controls]=\"ariaControls\" #buttonRef></button>\n}\n}", styles: [".hidden,.always-in-secondary-menu,.visibleXXSView,.visibleTinyView,.visibleSmallView,.visibleMediumView,.visibleLargeView,.visibleXLView,.visibleXXLView{display:none}\n"] }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PdfShyButtonComponent, decorators: [{ type: Component, args: [{ selector: 'pdf-shy-button', template: "@if (!onlySecondaryMenu) {\n@if (nestedContent) {\n<span (click)=\"onClick($event)\" onKeyDown=\"onClick($event)\" [class]=\"cssClass\">\n <ng-content></ng-content>\n</span>\n}\n@else {\n<button type=\"button\" [id]=\"primaryToolbarId\" class=\"toolbarButton\" [class]=\"cssClass\" [title]=\"title\"\n [attr.data-l10n-id]=\"l10nId\" [class.toggled]=\"toggled\" [disabled]=\"disabled\" (click)=\"onClick($event)\"\n [attr.aria-label]=\"title\" [attr.role]=\"role\" [attr.aria-expanded]=\"toggled ? 'true' : (ariaHasPopup ? 'false' : null)\"\n [attr.aria-haspopup]=\"ariaHasPopup\" [attr.aria-controls]=\"ariaControls\" #buttonRef></button>\n}\n}", styles: [".hidden,.always-in-secondary-menu,.visibleXXSView,.visibleTinyView,.visibleSmallView,.visibleMediumView,.visibleLargeView,.visibleXLView,.visibleXXLView{display:none}\n"] }] }], ctorParameters: () => [{ type: i1.PdfShyButtonService }, { type: i2.DomSanitizer }, { type: i0.Renderer2 }, { type: i3.PDFNotificationService }, { type: i4.PdfCspPolicyService }], propDecorators: { primaryToolbarId: [{ type: Input }], secondaryMenuId: [{ type: Input }], cssClass: [{ type: Input }], eventBusName: [{ type: Input }], l10nId: [{ type: Input }], l10nLabel: [{ type: Input }], title: [{ type: Input }], toggled: [{ type: Input }], disabled: [{ type: Input }], order: [{ type: Input }], action: [{ type: Input }], closeOnClick: [{ type: Input }], onlySecondaryMenu: [{ type: Input }], ariaHasPopup: [{ type: Input }], ariaControls: [{ type: Input }], role: [{ type: Input }], buttonRef: [{ type: ViewChild, args: ['buttonRef', { static: false }] }], nestedContent: [{ type: ContentChild, args: ['nestedContent', { static: false }] }], image: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pdf-shy-button.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-extended-pdf-viewer/src/lib/toolbar/pdf-shy-button/pdf-shy-button.component.ts","../../../../../../projects/ngx-extended-pdf-viewer/src/lib/toolbar/pdf-shy-button/pdf-shy-button.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterContentInit, AfterViewInit, Component, ContentChild, ElementRef, Input, OnChanges, OnInit, Renderer2, ViewChild, effect } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\nimport { IPDFViewerApplication } from '../../options/pdf-viewer-application';\nimport { PdfCspPolicyService } from '../../pdf-csp-policy.service';\nimport { PDFNotificationService } from '../../pdf-notification-service';\nimport { ResponsiveCSSClass } from '../../responsive-visibility';\nimport { PdfShyButtonService } from './pdf-shy-button-service';\n\n@Component({\n  selector: 'pdf-shy-button',\n  styleUrls: ['./pdf-shy-button.component.scss'],\n  templateUrl: './pdf-shy-button.component.html',\n})\nexport class PdfShyButtonComponent implements OnInit, OnChanges, AfterViewInit, AfterContentInit {\n  @Input()\n  public primaryToolbarId: string;\n\n  @Input()\n  public secondaryMenuId: string;\n\n  @Input()\n  public cssClass: ResponsiveCSSClass = 'invisible';\n\n  @Input()\n  public eventBusName: string | undefined = undefined;\n\n  @Input()\n  public l10nId: string;\n\n  @Input()\n  public l10nLabel: string;\n\n  @Input()\n  public title: string;\n\n  @Input()\n  public toggled: boolean;\n\n  @Input()\n  public disabled: boolean;\n\n  @Input()\n  public order: number;\n\n  @Input()\n  public action: ((htmlEvent?: Event, isSecondaryMenue?: boolean) => void) | undefined = undefined;\n\n  @Input()\n  public closeOnClick: boolean = true;\n\n  @Input()\n  public onlySecondaryMenu: boolean = false;\n\n  @Input()\n  public ariaHasPopup: boolean | 'true' | 'menu' | 'dialog' = false;\n\n  @Input()\n  public ariaControls: string | undefined;\n\n  @Input()\n  public role: string | undefined;\n\n  private PDFViewerApplication: IPDFViewerApplication | undefined;\n\n  public renderContent = false;\n\n  @ViewChild('buttonRef', { static: false }) buttonRef: ElementRef;\n\n  @ContentChild('nestedContent', { static: false }) nestedContent: ElementRef | null = null;\n\n  private _imageHtml: string | undefined;\n\n  public get imageHtml(): string | SafeHtml | undefined {\n    if (this._imageHtml) {\n      // allow non-literal svg tags (sanitized in the setter)\n      return this.sanitizer.bypassSecurityTrustHtml(this._imageHtml); // NOSONAR\n    }\n    return undefined;\n  }\n\n  @Input()\n  public set image(value: string) {\n    const svgTags = [\n      // 'a' is not allowed!\n      'animate',\n      'animateMotion',\n      'animateTransform',\n      'audio',\n      'canvas',\n      'circle',\n      'clipPath',\n      'defs',\n      'desc',\n      'discard',\n      'ellipse',\n      'feBlend',\n      'feColorMatrix',\n      'feComponentTransfer',\n      'feComposite',\n      'feConvolveMatrix',\n      'feDiffuseLighting',\n      'feDisplacementMap',\n      'feDistantLight',\n      'feDropShadow',\n      'feFlood',\n      'feFuncA',\n      'feFuncB',\n      'feFuncG',\n      'feFuncR',\n      'feGaussianBlur',\n      'feImage',\n      'feMerge',\n      'feMergeNode',\n      'feMorphology',\n      'feOffset',\n      'fePointLight',\n      'feSpecularLighting',\n      'feSpotLight',\n      'feTile',\n      'feTurbulence',\n      'filter',\n      'foreignObject',\n      'g',\n      'iframe',\n      'image',\n      'line',\n      'linearGradient',\n      'marker',\n      'mask',\n      'metadata',\n      'mpath',\n      'path',\n      'pattern',\n      'polygon',\n      'polyline',\n      'radialGradient',\n      'rect',\n      'script',\n      'set',\n      'stop',\n      'style',\n      'svg',\n      'switch',\n      'symbol',\n      'text',\n      'textPath',\n      'title',\n      'tspan',\n      'unknown',\n      'use',\n      'video',\n      'view',\n    ];\n\n    // only <svg> and SVG tags are allowed\n    const tags = value.split('<').filter((tag) => tag.length > 0);\n    const legal = tags.every((tag) => tag.startsWith('svg') || tag.startsWith('/') || svgTags.includes(tag.split(/\\s|>/)[0]));\n    if (!legal) {\n      throw new Error('Illegal image for PDFShyButton. Only SVG images are allowed. Please use only the tags <svg> and <path>. ' + value);\n    }\n    this._imageHtml = this.pdfCspPolicyService.sanitizeHTML(value);\n  }\n\n  constructor(\n    private pdfShyButtonServiceService: PdfShyButtonService,\n    private sanitizer: DomSanitizer,\n    private renderer: Renderer2,\n    notificationService: PDFNotificationService,\n    private pdfCspPolicyService: PdfCspPolicyService,\n  ) {\n    effect(() => {\n      this.PDFViewerApplication = notificationService.onPDFJSInitSignal();\n    });\n  }\n\n  public ngAfterViewInit(): void {\n    this.updateButtonImage();\n  }\n\n  public ngOnInit(): void {\n    this.pdfShyButtonServiceService.add(this);\n  }\n\n  public ngOnChanges(changes: any): void {\n    this.pdfShyButtonServiceService.update(this);\n  }\n\n  public onClick(htmlEvent: Event): void {\n    if (htmlEvent instanceof KeyboardEvent && htmlEvent.key !== 'Enter' && htmlEvent.key !== ' ') {\n      return;\n    }\n\n    if (this.action) {\n      this.action(htmlEvent, false);\n      htmlEvent.preventDefault();\n    } else if (this.eventBusName) {\n      this.PDFViewerApplication?.eventBus.dispatch(this.eventBusName);\n      htmlEvent.preventDefault();\n    }\n  }\n\n  public updateButtonImage() {\n    if (this.buttonRef) {\n      const el = this.buttonRef.nativeElement;\n      if (this._imageHtml) {\n        const temp = this.renderer.createElement('div');\n        this.pdfCspPolicyService.addTrustedHTML(temp, this._imageHtml);\n        const image = temp.children[0];\n        if (!el.innerHTML.includes(image.innerHTML)) {\n          // if using SSR, the HTML code may already be there\n          this.renderer.appendChild(el, image);\n        }\n      } else {\n        const childNodes = el.childNodes;\n        for (let child of childNodes) {\n          this.renderer.removeChild(el, child);\n        }\n      }\n    }\n  }\n\n  ngAfterContentInit() {\n    if (this.primaryToolbarId === 'nestedComponent') {\n      this.renderContent = !!this.nestedContent;\n      console.log('renderContent', this.renderContent);\n    }\n  }\n}\n","@if (!onlySecondaryMenu) {\n@if (nestedContent) {\n<span (click)=\"onClick($event)\" onKeyDown=\"onClick($event)\" [class]=\"cssClass\">\n  <ng-content></ng-content>\n</span>\n}\n@else {\n<button type=\"button\" [id]=\"primaryToolbarId\" class=\"toolbarButton\" [class]=\"cssClass\" [title]=\"title\"\n  [attr.data-l10n-id]=\"l10nId\" [class.toggled]=\"toggled\" [disabled]=\"disabled\" (click)=\"onClick($event)\"\n  [attr.aria-label]=\"title\" [attr.role]=\"role\" [attr.aria-expanded]=\"toggled ? 'true' : (ariaHasPopup ? 'false' : null)\"\n  [attr.aria-haspopup]=\"ariaHasPopup\" [attr.aria-controls]=\"ariaControls\" #buttonRef></button>\n}\n}"]}