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
JavaScript
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":"AAGA,MAAM,OAAO,oBAAoB;IAW/B,YAAoB,KAAa;QAAb,UAAK,GAAL,KAAK,CAAQ;QATzB,WAAM,GAAG,CAAC,CAAC;QACX,WAAM,GAAG,CAAC,CAAC;QACX,yBAAoB,GAAG,CAAC,CAAC;QACzB,eAAU,GAAG,CAAC,CAAC;QAOrB,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClE,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAChE,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE9D,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAEO,QAAQ;QACd,OAAO,CAAC,cAAc,IAAI,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,cAAc,GAAG,CAAC,CAAC,IAAI,CAAO,SAAU,CAAC,gBAAgB,GAAG,CAAC,CAAC,CAAC;IACjH,CAAC;IAEO,kBAAkB,CAAC,KAAiB;QAC1C,IAAI,CAAC,oBAAoB,GAAG,CAAC,CAAC;QAE9B,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE;YAC9B,MAAM,SAAS,GAAG,QAAQ,CAAC,cAAc,CAAC,iBAAiB,CAAmB,CAAC;YAC/E,MAAM,IAAI,GAAG,SAAS,CAAC,qBAAqB,EAAE,CAAC;YAC/C,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,CAAC,IAAI,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,EAAE;gBAC/E,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,GAAG,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,EAAE;oBACrH,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,CAAC,IAAI,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,EAAE;wBAC/E,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,GAAG,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,EAAE;4BACrH,IAAI,CAAC,MAAM,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;4BACpE,IAAI,CAAC,MAAM,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;4BACpE,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;4BAEzI,IAAI,KAAK,CAAC,UAAU,EAAE;gCACpB,KAAK,CAAC,cAAc,EAAE,CAAC;6BACxB;4BACD,KAAK,CAAC,eAAe,EAAE,CAAC;yBACzB;qBACF;iBACF;aACF;SACF;IACH,CAAC;IAEO,iBAAiB,CAAC,KAAiB;QACzC,MAAM,2BAA2B,GAAkC,MAAc,CAAC,2BAA2B,CAAC;QAC9G,MAAM,oBAAoB,GAAS,MAAc,CAAC,oBAAoB,CAAC;QAEvE,IAAI,IAAI,CAAC,oBAAoB,IAAI,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE;YAChE,OAAO;SACR;QAED,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;QACnI,MAAM,SAAS,GAAG,QAAQ,CAAC,cAAc,CAAC,iBAAiB,CAAmB,CAAC;QAC/E,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC,UAAU,CAAC;QACnD,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC,SAAS,CAAC;QAClD,IAAI,CAAC,UAAU,GAAG,aAAa,GAAG,IAAI,CAAC,oBAAoB,CAAC;QAC5D,IAAI,OAAO,GAAG,MAAM,CAAC,2BAA2B,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC;QACjE,IAAI,CAAC,OAAO,EAAE;YACZ,OAAO,GAAG,GAAG,CAAC;SACf;QAED,MAAM,WAAW,GAAG,oBAAoB,CAAC,SAAS,CAAC,aAAa,CAAC;QACjE,IAAI,WAAW,GAAG,IAAI,CAAC,UAAU,GAAG,OAAO,EAAE;YAC3C,IAAI,CAAC,UAAU,GAAG,OAAO,GAAG,WAAW,CAAC;SACzC;QACD,IAAI,OAAO,GAAG,MAAM,CAAC,2BAA2B,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC;QACjE,IAAI,CAAC,OAAO,EAAE;YACZ,OAAO,GAAG,EAAE,CAAC;SACd;QACD,IAAI,WAAW,GAAG,IAAI,CAAC,UAAU,GAAG,OAAO,EAAE;YAC3C,IAAI,CAAC,UAAU,GAAG,OAAO,GAAG,WAAW,CAAC;SACzC;QACD,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,GAAG,SAAS,IAAI,CAAC,UAAU,GAAG,CAAC;QAC1D,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,GAAG,GAAG,OAAO,MAAM,OAAO,IAAI,CAAC;QAEhE,IAAI,KAAK,CAAC,UAAU,EAAE;YACpB,KAAK,CAAC,cAAc,EAAE,CAAC;SACxB;QACD,KAAK,CAAC,eAAe,EAAE,CAAC;IAC1B,CAAC;IAEO,gBAAgB,CAAC,KAAiB;QACxC,MAAM,oBAAoB,GAAS,MAAc,CAAC,oBAAoB,CAAC;QACvE,IAAI,IAAI,CAAC,oBAAoB,IAAI,CAAC,EAAE;YAClC,OAAO;SACR;QACD,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,GAAG,MAAM,CAAC;QACrC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,GAAG,OAAO,CAAC;QAC5C,oBAAoB,CAAC,SAAS,CAAC,YAAY,IAAI,IAAI,CAAC,UAAU,CAAC;QAC/D,MAAM,SAAS,GAAG,QAAQ,CAAC,cAAc,CAAC,iBAAiB,CAAmB,CAAC;QAC/E,MAAM,IAAI,GAAG,SAAS,CAAC,qBAAqB,EAAE,CAAC;QAC/C,MAAM,EAAE,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC;QACnC,MAAM,EAAE,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC;QAClC,SAAS,CAAC,UAAU,IAAI,EAAE,GAAG,CAAC,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC,CAAC;QACnD,SAAS,CAAC,SAAS,IAAI,EAAE,GAAG,CAAC,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC,CAAC;QAClD,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAE5B,IAAI,KAAK,CAAC,UAAU,EAAE;YACpB,KAAK,CAAC,cAAc,EAAE,CAAC;SACxB;QACD,KAAK,CAAC,eAAe,EAAE,CAAC;IAC1B,CAAC;IAEO,oBAAoB;QAC1B,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,oBAAoB,GAAG,CAAC,CAAC;QAC1D,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;IACtB,CAAC;IAEM,mBAAmB;QACxB,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE;YACpB,OAAO;SACR;QACD,IAAI,CAAC,MAAM,GAAG,QAAQ,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC;QAChD,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,GAAG,EAAE;YAChC,QAAQ,CAAC,gBAAgB,CAAC,YAAY,EAAE,IAAI,CAAC,uBAAuB,CAAC,CAAC;YACtE,QAAQ,CAAC,gBAAgB,CAAC,WAAW,EAAE,IAAI,CAAC,sBAAsB,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YACxF,QAAQ,CAAC,gBAAgB,CAAC,UAAU,EAAE,IAAI,CAAC,qBAAqB,CAAC,CAAC;QACpE,CAAC,CAAC,CAAC;IACL,CAAC;IAEM,gBAAgB;QACrB,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE;YACpB,OAAO;SACR;QACD,QAAQ,CAAC,mBAAmB,CAAC,YAAY,EAAE,IAAI,CAAC,uBAAuB,CAAC,CAAC;QACzE,QAAQ,CAAC,mBAAmB,CAAC,WAAW,EAAE,IAAI,CAAC,sBAAsB,CAAC,CAAC;QACvE,QAAQ,CAAC,mBAAmB,CAAC,UAAU,EAAE,IAAI,CAAC,qBAAqB,CAAC,CAAC;IACvE,CAAC;CACF","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"]}