UNPKG

@herdwatch/ngx-ionic-image-viewer

Version:

The angular workspace to develop the ngx-ionic-image-viewer component

232 lines 35.8 kB
import { Component, Input, ViewChild, CUSTOM_ELEMENTS_SCHEMA } from '@angular/core'; import { IonHeader, IonToolbar, IonButton, IonButtons, IonIcon, IonTitle, IonContent, IonFooter, IonText } from '@ionic/angular/standalone'; import { NgClass } from '@angular/common'; import { addIcons } from 'ionicons'; import { close } from 'ionicons/icons'; import * as i0 from "@angular/core"; import * as i1 from "@ionic/angular/standalone"; export class ViewerModalComponent { // @ViewChild('sliderRef', { static: true }) slides!: IonSlides; constructor(modalController) { this.modalController = modalController; this.alt = ''; this.scheme = 'auto'; this.slideOptions = {}; this.srcFallback = ''; this.srcHighRes = ''; this.swipeToClose = true; this.text = ''; this.title = ''; this.titleSize = ''; this.defaultSlideOptions = { zoom: { enabled: true, }, }; this.options = {}; this.swipeState = { phase: 'init', direction: 'none', swipeType: 'none', startX: 0, startY: 0, distance: 0, distanceX: 0, distanceY: 0, threshold: 150, // required min distance traveled to be considered swipe restraint: 100, // maximum distance allowed at the same time in perpendicular direction allowedTime: 500, // maximum time allowed to travel that distance elapsedTime: 0, startTime: 0, }; addIcons({ close }); } async ngOnInit() { this.options = { ...this.defaultSlideOptions, ...this.slideOptions }; this.src = this.srcHighRes || this.src; this.setStyle(); this.setScheme(this.scheme); this.initSwipeToClose(this.swipeToClose); /** * Current Workaround * See reported bug: https://github.com/ionic-team/ionic/issues/19638#issuecomment-584828315 * Hint: Comment in '<ion-slide>' in component */ } setStyle() { const el = document.querySelector('.ion-img-viewer'); el?.style.setProperty('--height', '100%'); el?.style.setProperty('--width', '100%'); el?.style.setProperty('--border-radius', '0'); } setScheme(scheme) { if (scheme && scheme === 'auto') { return; } const el = document.querySelector('.ion-img-viewer'); if (this.scheme === 'light') { el?.style.setProperty('--ion-background-color', '#ffffff'); el?.style.setProperty('--ion-background-color-rgb', '255, 255, 255'); el?.style.setProperty('--ion-text-color', '#000'); el?.style.setProperty('--ion-text-color-rgb', '0,0,0'); } if (this.scheme === 'dark') { if (el?.classList.contains('ios')) { el?.style.setProperty('--ion-background-color', '#000000'); el?.style.setProperty('--ion-background-color-rgb', '0, 0, 0'); } else { el?.style.setProperty('--ion-background-color', '#121212'); el?.style.setProperty('--ion-background-color-rgb', '18,18,18'); } el?.style.setProperty('--ion-text-color', '#ffffff'); el?.style.setProperty('--ion-text-color-rgb', '255,255,255'); } } /** * @see http://www.javascriptkit.com/javatutors/touchevents3.shtml */ initSwipeToClose(isActive = true) { if (!isActive) { return; } const el = document.querySelector('ion-modal'); el?.addEventListener('mousedown', (event) => this.swipeStart(event), true); el?.addEventListener('mousemove', (event) => this.swipeMove(event), true); el?.addEventListener('mouseup', () => this.swipeEnd(), true); el?.addEventListener('touchstart', (event) => this.swipeStart(event), true); el?.addEventListener('touchmove', (event) => this.swipeMove(event), true); el?.addEventListener('touchend', () => this.swipeEnd(), true); this.modalController.getTop().then((modal) => { modal?.onWillDismiss().then(() => { document.removeEventListener('mousedown', this.swipeStart, true); document.removeEventListener('mousemove', this.swipeMove, true); document.removeEventListener('mouseup', this.swipeMove, true); document.removeEventListener('touchstart', this.swipeStart, true); document.removeEventListener('touchmove', this.swipeMove, true); document.removeEventListener('touchend', this.swipeMove, true); }); }); } // eslint-disable-next-line @typescript-eslint/no-explicit-any swipeStart(event) { const { pageX, pageY } = event.type === 'touchstart' && event?.changedTouches ? event.changedTouches[0] : event; this.swipeState = { ...this.swipeState, phase: 'start', direction: 'none', distance: 0, startX: pageX, startY: pageY, startTime: new Date().getTime(), }; } // eslint-disable-next-line @typescript-eslint/no-explicit-any swipeMove(event) { const { pageX, pageY } = event.type === 'touchmove' && event?.changedTouches ? event.changedTouches[0] : event; // get horizontal dist traveled by finger while in contact with surface const distanceX = pageX - this.swipeState.startX; // get vertical dist traveled by finger while in contact with surface const distanceY = pageY - this.swipeState.startY; let direction; let distance; if (Math.abs(distanceX) > Math.abs(distanceY)) { // if distance traveled horizontally is greater than vertically, consider this a horizontal swipe direction = distanceX < 0 ? 'left' : 'right'; distance = distanceX; } else { // else consider this a vertical swipe direction = distanceY < 0 ? 'up' : 'down'; distance = distanceY; } this.swipeState = { ...this.swipeState, phase: 'move', direction, distance, distanceX, distanceY, }; if (event.cancelable) { event.preventDefault(); } } swipeEnd() { if (this.swipeState.phase === 'none') { return; } const { allowedTime, direction, restraint, startTime, threshold, distanceX, distanceY, } = this.swipeState; let swipeType = null; const elapsedTime = new Date().getTime() - startTime; // get time elapsed if (elapsedTime <= allowedTime) { // first condition for a swipe met if (Math.abs(distanceX) >= threshold && Math.abs(distanceY) <= restraint) { // 2nd condition for horizontal swipe met swipeType = direction; // set swipeType to either "left" or "right" } else if (Math.abs(distanceY) >= threshold && Math.abs(distanceX) <= restraint) { // 2nd condition for vertical swipe met swipeType = direction; // set swipeType to either "top" or "down" } } this.swipeState = { ...this.swipeState, phase: 'end', swipeType: swipeType ?? '', }; if (swipeType === 'down') { return this.closeModal(); } } closeModal() { this.modalController.dismiss(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.10", ngImport: i0, type: ViewerModalComponent, deps: [{ token: i1.ModalController }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.10", type: ViewerModalComponent, isStandalone: true, selector: "ion-viewer-modal", inputs: { alt: "alt", scheme: "scheme", slideOptions: "slideOptions", src: "src", srcFallback: "srcFallback", srcHighRes: "srcHighRes", swipeToClose: "swipeToClose", text: "text", title: "title", titleSize: "titleSize" }, viewQueries: [{ propertyName: "swiperRef", first: true, predicate: ["swiper"], descendants: true }], ngImport: i0, template: "<ion-header [ngClass]=\"{ 'no-title': title && title.length <= 0 }\">\n <ion-toolbar>\n <ion-buttons slot=\"primary\">\n <ion-button data-cy=\"closeModal\" (click)=\"closeModal()\">\n <ion-icon slot=\"icon-only\" name=\"close\"></ion-icon>\n </ion-button>\n </ion-buttons>\n <ion-title [size]=\"titleSize\">{{ title }}</ion-title>\n </ion-toolbar>\n</ion-header>\n\n<ion-content [forceOverscroll]=\"false\">\n <swiper-container [options]=\"options\" centeredSlides=\"true\" passiveListeners=\"false\" zoom=\"true\" #swiper>\n <swiper-slide>\n <div class=\"swiper-zoom-container\">\n <img [alt]=\"this.alt\" [src]=\"this.src\"/>\n </div>\n </swiper-slide>\n </swiper-container>\n</ion-content>\n\n<ion-footer [ngClass]=\"{ 'no-text': text && text.length <= 0 }\">\n <ion-toolbar class=\"ion-text-center\">\n <ion-text>{{ text }}</ion-text>\n </ion-toolbar>\n</ion-footer>\n", styles: ["ion-slides{--height: 100%;height:100%}ion-toolbar{--border-style: none;--background: rgba(var(--ion-background-color-rgb, (255, 255, 255)), .6);background:rgba(var(--ion-background-color-rgb, 255, 255, 255),.6)}ion-header{opacity:1;position:absolute}ion-header.no-title:after{content:none}ion-header.no-title ion-toolbar{--background: rgba(0, 0, 0, 0);background:#0000}ion-footer{position:absolute;bottom:0}ion-footer.no-text:before{content:none}ion-footer.no-text ion-toolbar{--background: rgba(0, 0, 0, 0);background:#0000}swiper-container{width:100%;height:100%}swiper-slide{overflow:hidden}\n"], dependencies: [{ kind: "component", type: IonHeader, selector: "ion-header", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonTitle, selector: "ion-title", inputs: ["color", "size"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }, { kind: "component", type: IonFooter, selector: "ion-footer", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.10", ngImport: i0, type: ViewerModalComponent, decorators: [{ type: Component, args: [{ selector: 'ion-viewer-modal', standalone: true, schemas: [CUSTOM_ELEMENTS_SCHEMA], imports: [ IonHeader, IonToolbar, IonButton, NgClass, IonButtons, IonIcon, IonTitle, IonContent, IonFooter, IonText, ], template: "<ion-header [ngClass]=\"{ 'no-title': title && title.length <= 0 }\">\n <ion-toolbar>\n <ion-buttons slot=\"primary\">\n <ion-button data-cy=\"closeModal\" (click)=\"closeModal()\">\n <ion-icon slot=\"icon-only\" name=\"close\"></ion-icon>\n </ion-button>\n </ion-buttons>\n <ion-title [size]=\"titleSize\">{{ title }}</ion-title>\n </ion-toolbar>\n</ion-header>\n\n<ion-content [forceOverscroll]=\"false\">\n <swiper-container [options]=\"options\" centeredSlides=\"true\" passiveListeners=\"false\" zoom=\"true\" #swiper>\n <swiper-slide>\n <div class=\"swiper-zoom-container\">\n <img [alt]=\"this.alt\" [src]=\"this.src\"/>\n </div>\n </swiper-slide>\n </swiper-container>\n</ion-content>\n\n<ion-footer [ngClass]=\"{ 'no-text': text && text.length <= 0 }\">\n <ion-toolbar class=\"ion-text-center\">\n <ion-text>{{ text }}</ion-text>\n </ion-toolbar>\n</ion-footer>\n", styles: ["ion-slides{--height: 100%;height:100%}ion-toolbar{--border-style: none;--background: rgba(var(--ion-background-color-rgb, (255, 255, 255)), .6);background:rgba(var(--ion-background-color-rgb, 255, 255, 255),.6)}ion-header{opacity:1;position:absolute}ion-header.no-title:after{content:none}ion-header.no-title ion-toolbar{--background: rgba(0, 0, 0, 0);background:#0000}ion-footer{position:absolute;bottom:0}ion-footer.no-text:before{content:none}ion-footer.no-text ion-toolbar{--background: rgba(0, 0, 0, 0);background:#0000}swiper-container{width:100%;height:100%}swiper-slide{overflow:hidden}\n"] }] }], ctorParameters: () => [{ type: i1.ModalController }], propDecorators: { alt: [{ type: Input }], scheme: [{ type: Input }], slideOptions: [{ type: Input }], src: [{ type: Input }], srcFallback: [{ type: Input }], srcHighRes: [{ type: Input }], swipeToClose: [{ type: Input }], text: [{ type: Input }], title: [{ type: Input }], titleSize: [{ type: Input }], swiperRef: [{ type: ViewChild, args: ['swiper'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"viewer-modal.component.js","sourceRoot":"","sources":["../../../../../../libs/ngx-ionic-image-viewer/src/lib/viewer-modal/viewer-modal.component.ts","../../../../../../libs/ngx-ionic-image-viewer/src/lib/viewer-modal/viewer-modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, Input, ViewChild, ElementRef, CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';\nimport { ModalController } from '@ionic/angular/standalone';\nimport { IonHeader, IonToolbar, IonButton, IonButtons, IonIcon, IonTitle, IonContent, IonFooter, IonText } from '@ionic/angular/standalone';\nimport { NgClass } from '@angular/common';\nimport { addIcons } from 'ionicons';\nimport { close } from 'ionicons/icons';\n@Component({\n  selector: 'ion-viewer-modal',\n  templateUrl: './viewer-modal.component.html',\n  styleUrls: ['./viewer-modal.component.scss'],\n  standalone: true,\n  schemas: [CUSTOM_ELEMENTS_SCHEMA],\n  imports: [\n    IonHeader,\n    IonToolbar,\n    IonButton,\n    NgClass,\n    IonButtons,\n    IonIcon,\n    IonTitle,\n    IonContent,\n    IonFooter,\n    IonText,\n  ],\n})\nexport class ViewerModalComponent implements OnInit {\n  @Input() alt?: string = '';\n  @Input() scheme?: string = 'auto';\n  @Input() slideOptions?: object = {};\n  @Input() src!: string;\n  @Input() srcFallback?: string = '';\n  @Input() srcHighRes?: string = '';\n  @Input() swipeToClose?: boolean = true;\n  @Input() text?: string = '';\n  @Input() title?: string = '';\n  @Input() titleSize?: string = '';\n\n  defaultSlideOptions = {\n    zoom: {\n      enabled: true,\n    },\n  };\n\n  options = {};\n\n  swipeState = {\n    phase: 'init',\n    direction: 'none',\n    swipeType: 'none',\n    startX: 0,\n    startY: 0,\n    distance: 0,\n    distanceX: 0,\n    distanceY: 0,\n    threshold: 150, // required min distance traveled to be considered swipe\n    restraint: 100, // maximum distance allowed at the same time in perpendicular direction\n    allowedTime: 500, // maximum time allowed to travel that distance\n    elapsedTime: 0,\n    startTime: 0,\n  };\n\n  @ViewChild('swiper') swiperRef: ElementRef | undefined;\n\n  // @ViewChild('sliderRef', { static: true }) slides!: IonSlides;\n\n  constructor(private modalController: ModalController) {\n    addIcons({close})\n  }\n\n  async ngOnInit() {\n    this.options = { ...this.defaultSlideOptions, ...this.slideOptions };\n    this.src = this.srcHighRes || this.src;\n    this.setStyle();\n    this.setScheme(this.scheme);\n    this.initSwipeToClose(this.swipeToClose);\n\n    /**\n     * Current Workaround\n     * See reported bug: https://github.com/ionic-team/ionic/issues/19638#issuecomment-584828315\n     * Hint: Comment in '<ion-slide>' in component\n     */\n  }\n\n  setStyle() {\n    const el: HTMLElement | null = document.querySelector('.ion-img-viewer');\n    el?.style.setProperty('--height', '100%');\n    el?.style.setProperty('--width', '100%');\n    el?.style.setProperty('--border-radius', '0');\n  }\n\n  setScheme(scheme: string | undefined) {\n    if (scheme && scheme === 'auto') {\n      return;\n    }\n\n    const el: HTMLElement | null = document.querySelector('.ion-img-viewer');\n\n    if (this.scheme === 'light') {\n      el?.style.setProperty('--ion-background-color', '#ffffff');\n      el?.style.setProperty('--ion-background-color-rgb', '255, 255, 255');\n      el?.style.setProperty('--ion-text-color', '#000');\n      el?.style.setProperty('--ion-text-color-rgb', '0,0,0');\n    }\n\n    if (this.scheme === 'dark') {\n      if (el?.classList.contains('ios')) {\n        el?.style.setProperty('--ion-background-color', '#000000');\n        el?.style.setProperty('--ion-background-color-rgb', '0, 0, 0');\n      } else {\n        el?.style.setProperty('--ion-background-color', '#121212');\n        el?.style.setProperty('--ion-background-color-rgb', '18,18,18');\n      }\n      el?.style.setProperty('--ion-text-color', '#ffffff');\n      el?.style.setProperty('--ion-text-color-rgb', '255,255,255');\n    }\n  }\n\n  /**\n   * @see http://www.javascriptkit.com/javatutors/touchevents3.shtml\n   */\n  initSwipeToClose(isActive = true) {\n    if (!isActive) {\n      return;\n    }\n\n    const el = document.querySelector('ion-modal');\n    el?.addEventListener('mousedown', (event) => this.swipeStart(event), true);\n    el?.addEventListener('mousemove', (event) => this.swipeMove(event), true);\n    el?.addEventListener('mouseup', () => this.swipeEnd(), true);\n    el?.addEventListener('touchstart', (event) => this.swipeStart(event), true);\n    el?.addEventListener('touchmove', (event) => this.swipeMove(event), true);\n    el?.addEventListener('touchend', () => this.swipeEnd(), true);\n\n    this.modalController.getTop().then((modal) => {\n      modal?.onWillDismiss().then(() => {\n        document.removeEventListener('mousedown', this.swipeStart, true);\n        document.removeEventListener('mousemove', this.swipeMove, true);\n        document.removeEventListener('mouseup', this.swipeMove, true);\n        document.removeEventListener('touchstart', this.swipeStart, true);\n        document.removeEventListener('touchmove', this.swipeMove, true);\n        document.removeEventListener('touchend', this.swipeMove, true);\n      });\n    });\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  swipeStart(event: any) {\n    const { pageX, pageY } =\n      event.type === 'touchstart' && event?.changedTouches\n        ? event.changedTouches[0]\n        : event;\n\n    this.swipeState = {\n      ...this.swipeState,\n      phase: 'start',\n      direction: 'none',\n      distance: 0,\n      startX: pageX,\n      startY: pageY,\n      startTime: new Date().getTime(),\n    };\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  swipeMove(event: any) {\n    const { pageX, pageY } =\n      event.type === 'touchmove' && event?.changedTouches\n        ? event.changedTouches[0]\n        : event;\n    // get horizontal dist traveled by finger while in contact with surface\n    const distanceX = pageX - this.swipeState.startX;\n    // get vertical dist traveled by finger while in contact with surface\n    const distanceY = pageY - this.swipeState.startY;\n    let direction;\n    let distance;\n\n    if (Math.abs(distanceX) > Math.abs(distanceY)) {\n      // if distance traveled horizontally is greater than vertically, consider this a horizontal swipe\n      direction = distanceX < 0 ? 'left' : 'right';\n      distance = distanceX;\n    } else {\n      // else consider this a vertical swipe\n      direction = distanceY < 0 ? 'up' : 'down';\n      distance = distanceY;\n    }\n    this.swipeState = {\n      ...this.swipeState,\n      phase: 'move',\n      direction,\n      distance,\n      distanceX,\n      distanceY,\n    };\n    if (event.cancelable) {\n      event.preventDefault();\n    }\n  }\n\n  swipeEnd() {\n    if (this.swipeState.phase === 'none') {\n      return;\n    }\n    const {\n      allowedTime,\n      direction,\n      restraint,\n      startTime,\n      threshold,\n      distanceX,\n      distanceY,\n    } = this.swipeState;\n    let swipeType = null;\n\n    const elapsedTime = new Date().getTime() - startTime; // get time elapsed\n    if (elapsedTime <= allowedTime) {\n      // first condition for a swipe met\n      if (\n        Math.abs(distanceX) >= threshold &&\n        Math.abs(distanceY) <= restraint\n      ) {\n        // 2nd condition for horizontal swipe met\n        swipeType = direction; // set swipeType to either \"left\" or \"right\"\n      } else if (\n        Math.abs(distanceY) >= threshold &&\n        Math.abs(distanceX) <= restraint\n      ) {\n        // 2nd condition for vertical swipe met\n        swipeType = direction; // set swipeType to either \"top\" or \"down\"\n      }\n    }\n\n    this.swipeState = {\n      ...this.swipeState,\n      phase: 'end',\n      swipeType: swipeType ?? '',\n    };\n\n    if (swipeType === 'down') {\n      return this.closeModal();\n    }\n  }\n\n  closeModal() {\n    this.modalController.dismiss();\n  }\n}\n","<ion-header [ngClass]=\"{ 'no-title': title && title.length <= 0 }\">\n  <ion-toolbar>\n    <ion-buttons slot=\"primary\">\n      <ion-button data-cy=\"closeModal\" (click)=\"closeModal()\">\n        <ion-icon slot=\"icon-only\" name=\"close\"></ion-icon>\n      </ion-button>\n    </ion-buttons>\n    <ion-title [size]=\"titleSize\">{{ title }}</ion-title>\n  </ion-toolbar>\n</ion-header>\n\n<ion-content [forceOverscroll]=\"false\">\n  <swiper-container [options]=\"options\" centeredSlides=\"true\" passiveListeners=\"false\" zoom=\"true\" #swiper>\n    <swiper-slide>\n      <div class=\"swiper-zoom-container\">\n        <img [alt]=\"this.alt\" [src]=\"this.src\"/>\n      </div>\n    </swiper-slide>\n  </swiper-container>\n</ion-content>\n\n<ion-footer [ngClass]=\"{ 'no-text': text && text.length <= 0 }\">\n  <ion-toolbar class=\"ion-text-center\">\n    <ion-text>{{ text }}</ion-text>\n  </ion-toolbar>\n</ion-footer>\n"]}