@herdwatch/ngx-ionic-image-viewer
Version:
The angular workspace to develop the ngx-ionic-image-viewer component
232 lines • 35.8 kB
JavaScript
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"]}