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.

115 lines 19.8 kB
export class PinchOnMobileSupport { constructor(_zone) { this._zone = _zone; this.startX = 0; this.startY = 0; this.initialPinchDistance = 0; this.pinchScale = 1; this.boundOnViewerTouchStart = this.onViewerTouchStart.bind(this); this.boundOnViewerTouchMove = this.onViewerTouchMove.bind(this); this.boundOnViewerTouchEnd = this.onViewerTouchEnd.bind(this); this.initializePinchZoom(); } isMobile() { return ('ontouchstart' in window) || (navigator.maxTouchPoints > 0) || (navigator.msMaxTouchPoints > 0); } onViewerTouchStart(event) { this.initialPinchDistance = 0; if (event.touches.length === 2) { const container = document.getElementById('viewerContainer'); const rect = container.getBoundingClientRect(); if (event.touches[0].pageX >= rect.left && event.touches[0].pageX <= rect.right) { if (event.touches[0].pageY >= (rect.top + window.scrollY) && event.touches[0].pageY <= (rect.bottom + window.scrollY)) { if (event.touches[1].pageX >= rect.left && event.touches[1].pageX <= rect.right) { if (event.touches[1].pageY >= (rect.top + window.scrollY) && event.touches[1].pageY <= (rect.bottom + window.scrollY)) { this.startX = (event.touches[0].pageX + event.touches[1].pageX) / 2; this.startY = (event.touches[0].pageY + event.touches[1].pageY) / 2; this.initialPinchDistance = Math.hypot(event.touches[1].pageX - event.touches[0].pageX, event.touches[1].pageY - event.touches[0].pageY); if (event.cancelable) { event.preventDefault(); } event.stopPropagation(); } } } } } } onViewerTouchMove(event) { const PDFViewerApplicationOptions = window.PDFViewerApplicationOptions; const PDFViewerApplication = window.PDFViewerApplication; if (this.initialPinchDistance <= 0 || event.touches.length !== 2) { return; } const pinchDistance = Math.hypot(event.touches[1].pageX - event.touches[0].pageX, event.touches[1].pageY - event.touches[0].pageY); const container = document.getElementById('viewerContainer'); const originX = this.startX + container.scrollLeft; const originY = this.startY + container.scrollTop; this.pinchScale = pinchDistance / this.initialPinchDistance; let minZoom = Number(PDFViewerApplicationOptions.get('minZoom')); if (!minZoom) { minZoom = 0.1; } const currentZoom = PDFViewerApplication.pdfViewer._currentScale; if (currentZoom * this.pinchScale < minZoom) { this.pinchScale = minZoom / currentZoom; } let maxZoom = Number(PDFViewerApplicationOptions.get('maxZoom')); if (!maxZoom) { maxZoom = 10; } if (currentZoom * this.pinchScale > maxZoom) { this.pinchScale = maxZoom / currentZoom; } this.viewer.style.transform = `scale(${this.pinchScale})`; this.viewer.style.transformOrigin = `${originX}px ${originY}px`; if (event.cancelable) { event.preventDefault(); } event.stopPropagation(); } onViewerTouchEnd(event) { const PDFViewerApplication = window.PDFViewerApplication; if (this.initialPinchDistance <= 0) { return; } this.viewer.style.transform = `none`; this.viewer.style.transformOrigin = `unset`; PDFViewerApplication.pdfViewer.currentScale *= this.pinchScale; const container = document.getElementById('viewerContainer'); const rect = container.getBoundingClientRect(); const dx = this.startX - rect.left; const dy = this.startY - rect.top; container.scrollLeft += dx * (this.pinchScale - 1); container.scrollTop += dy * (this.pinchScale - 1); this.resetPinchZoomParams(); if (event.cancelable) { event.preventDefault(); } event.stopPropagation(); } resetPinchZoomParams() { this.startX = this.startY = this.initialPinchDistance = 0; this.pinchScale = 1; } initializePinchZoom() { if (!this.isMobile()) { return; } this.viewer = document.getElementById('viewer'); this._zone.runOutsideAngular(() => { document.addEventListener('touchstart', this.boundOnViewerTouchStart); document.addEventListener('touchmove', this.boundOnViewerTouchMove, { passive: false }); document.addEventListener('touchend', this.boundOnViewerTouchEnd); }); } destroyPinchZoom() { if (!this.isMobile()) { return; } document.removeEventListener('touchstart', this.boundOnViewerTouchStart); document.removeEventListener('touchmove', this.boundOnViewerTouchMove); document.removeEventListener('touchend', this.boundOnViewerTouchEnd); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pinch-on-mobile-support.js","sourceRoot":"","sources":["../../../../projects/ngx-extended-pdf-viewer/src/lib/pinch-on-mobile-support.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgZone } from '@angular/core';\nimport { IPDFViewerApplicationOptions } from './options/pdf-viewer-application-options';\n\nexport class PinchOnMobileSupport {\n  private viewer: any;\n  private startX = 0;\n  private startY = 0;\n  private initialPinchDistance = 0;\n  private pinchScale = 1;\n\n  private boundOnViewerTouchStart: any;\n  private boundOnViewerTouchMove: any;\n  private boundOnViewerTouchEnd: any;\n\n  constructor(private _zone: NgZone) {\n    this.boundOnViewerTouchStart = this.onViewerTouchStart.bind(this);\n    this.boundOnViewerTouchMove = this.onViewerTouchMove.bind(this);\n    this.boundOnViewerTouchEnd = this.onViewerTouchEnd.bind(this);\n\n    this.initializePinchZoom();\n  }\n\n  private isMobile() {\n    return ('ontouchstart' in window) || (navigator.maxTouchPoints > 0) || ((<any>navigator).msMaxTouchPoints > 0);\n  }\n\n  private onViewerTouchStart(event: TouchEvent): void {\n    this.initialPinchDistance = 0;\n\n    if (event.touches.length === 2) {\n      const container = document.getElementById('viewerContainer') as HTMLDivElement;\n      const rect = container.getBoundingClientRect();\n      if (event.touches[0].pageX >= rect.left && event.touches[0].pageX <= rect.right) {\n        if (event.touches[0].pageY >= (rect.top + window.scrollY) && event.touches[0].pageY <= (rect.bottom + window.scrollY)) {\n          if (event.touches[1].pageX >= rect.left && event.touches[1].pageX <= rect.right) {\n            if (event.touches[1].pageY >= (rect.top + window.scrollY) && event.touches[1].pageY <= (rect.bottom + window.scrollY)) {\n              this.startX = (event.touches[0].pageX + event.touches[1].pageX) / 2;\n              this.startY = (event.touches[0].pageY + event.touches[1].pageY) / 2;\n              this.initialPinchDistance = Math.hypot(event.touches[1].pageX - event.touches[0].pageX, event.touches[1].pageY - event.touches[0].pageY);\n\n              if (event.cancelable) {\n                event.preventDefault();\n              }\n              event.stopPropagation();\n            }\n          }\n        }\n      }\n    }\n  }\n\n  private onViewerTouchMove(event: TouchEvent): void {\n    const PDFViewerApplicationOptions: IPDFViewerApplicationOptions = (window as any).PDFViewerApplicationOptions;\n    const PDFViewerApplication: any = (window as any).PDFViewerApplication;\n\n    if (this.initialPinchDistance <= 0 || event.touches.length !== 2) {\n      return;\n    }\n\n    const pinchDistance = Math.hypot(event.touches[1].pageX - event.touches[0].pageX, event.touches[1].pageY - event.touches[0].pageY);\n    const container = document.getElementById('viewerContainer') as HTMLDivElement;\n    const originX = this.startX + container.scrollLeft;\n    const originY = this.startY + container.scrollTop;\n    this.pinchScale = pinchDistance / this.initialPinchDistance;\n    let minZoom = Number(PDFViewerApplicationOptions.get('minZoom'));\n    if (!minZoom) {\n      minZoom = 0.1;\n    }\n\n    const currentZoom = PDFViewerApplication.pdfViewer._currentScale;\n    if (currentZoom * this.pinchScale < minZoom) {\n      this.pinchScale = minZoom / currentZoom;\n    }\n    let maxZoom = Number(PDFViewerApplicationOptions.get('maxZoom'));\n    if (!maxZoom) {\n      maxZoom = 10;\n    }\n    if (currentZoom * this.pinchScale > maxZoom) {\n      this.pinchScale = maxZoom / currentZoom;\n    }\n    this.viewer.style.transform = `scale(${this.pinchScale})`;\n    this.viewer.style.transformOrigin = `${originX}px ${originY}px`;\n\n    if (event.cancelable) {\n      event.preventDefault();\n    }\n    event.stopPropagation();\n  }\n\n  private onViewerTouchEnd(event: TouchEvent): void {\n    const PDFViewerApplication: any = (window as any).PDFViewerApplication;\n    if (this.initialPinchDistance <= 0) {\n      return;\n    }\n    this.viewer.style.transform = `none`;\n    this.viewer.style.transformOrigin = `unset`;\n    PDFViewerApplication.pdfViewer.currentScale *= this.pinchScale;\n    const container = document.getElementById('viewerContainer') as HTMLDivElement;\n    const rect = container.getBoundingClientRect();\n    const dx = this.startX - rect.left;\n    const dy = this.startY - rect.top;\n    container.scrollLeft += dx * (this.pinchScale - 1);\n    container.scrollTop += dy * (this.pinchScale - 1);\n    this.resetPinchZoomParams();\n\n    if (event.cancelable) {\n      event.preventDefault();\n    }\n    event.stopPropagation();\n  }\n\n  private resetPinchZoomParams(): void {\n    this.startX = this.startY = this.initialPinchDistance = 0;\n    this.pinchScale = 1;\n  }\n\n  public initializePinchZoom(): void {\n    if (!this.isMobile()) {\n      return;\n    }\n    this.viewer = document.getElementById('viewer');\n    this._zone.runOutsideAngular(() => {\n      document.addEventListener('touchstart', this.boundOnViewerTouchStart);\n      document.addEventListener('touchmove', this.boundOnViewerTouchMove, { passive: false });\n      document.addEventListener('touchend', this.boundOnViewerTouchEnd);\n    });\n  }\n\n  public destroyPinchZoom(): void {\n    if (!this.isMobile()) {\n      return;\n    }\n    document.removeEventListener('touchstart', this.boundOnViewerTouchStart);\n    document.removeEventListener('touchmove', this.boundOnViewerTouchMove);\n    document.removeEventListener('touchend', this.boundOnViewerTouchEnd);\n  }\n}\n"]}