├── src ├── assets │ └── .gitkeep ├── favicon.ico ├── environments │ ├── environment.prod.ts │ └── environment.ts ├── styles.sass ├── app │ ├── app-routing.module.ts │ ├── app.component.ts │ ├── app.module.ts │ ├── app.component.spec.ts │ ├── app.component.sass │ └── app.component.html ├── index.html ├── main.ts ├── test.ts └── polyfills.ts ├── projects └── ngx-pinch-zoom │ ├── src │ ├── public-api.ts │ ├── lib │ │ ├── pinch-zoom.component.html │ │ ├── interfaces.ts │ │ ├── properties.ts │ │ ├── pinch-zoom.component.sass │ │ ├── pinch-zoom.component.ts │ │ ├── touches.ts │ │ └── ivypinch.ts │ └── test.ts │ ├── cypress │ ├── plugins │ │ └── index.ts │ ├── tsconfig.json │ ├── integration │ │ └── spec.ts │ └── support │ │ ├── index.ts │ │ └── commands.ts │ ├── ng-package.json │ ├── tsconfig.lib.prod.json │ ├── tsconfig.spec.json │ ├── cypress.json │ ├── tsconfig.lib.json │ ├── package.json │ ├── karma.conf.js │ ├── .eslintrc.js │ └── README.md ├── cypress ├── fixtures │ └── example.json ├── plugins │ └── index.ts ├── tsconfig.json ├── integration │ └── spec.ts └── support │ ├── index.ts │ └── commands.ts ├── .prettierrc ├── .prettierignore ├── tsconfig.app.json ├── .editorconfig ├── cypress.json ├── tsconfig.spec.json ├── .gitignore ├── tsconfig.json ├── LICENSE ├── karma.conf.js ├── package.json ├── angular.json ├── .eslintrc.js └── README.md /src/assets/.gitkeep: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /src/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/medDV-GmbH/ngx-pinch-zoom/HEAD/src/favicon.ico -------------------------------------------------------------------------------- /src/environments/environment.prod.ts: -------------------------------------------------------------------------------- 1 | export const environment = { 2 | production: true, 3 | }; 4 | -------------------------------------------------------------------------------- /src/styles.sass: -------------------------------------------------------------------------------- 1 | /* You can add global styles to this file, and also import other style files */ 2 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/public-api.ts: -------------------------------------------------------------------------------- 1 | /* 2 | * Public API Surface of ngx-pinch-zoom 3 | */ 4 | 5 | export * from './lib/interfaces'; 6 | export * from './lib/pinch-zoom.component'; 7 | -------------------------------------------------------------------------------- /cypress/fixtures/example.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Using fixtures to represent data", 3 | "email": "hello@cypress.io", 4 | "body": "Fixtures are a great way to mock data for responses to routes" 5 | } 6 | -------------------------------------------------------------------------------- /cypress/plugins/index.ts: -------------------------------------------------------------------------------- 1 | // Plugins enable you to tap into, modify, or extend the internal behavior of Cypress 2 | // For more info, visit https://on.cypress.io/plugins-api 3 | module.exports = (on, config) => {}; 4 | -------------------------------------------------------------------------------- /cypress/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "../tsconfig.json", 3 | "include": ["**/*.ts"], 4 | "compilerOptions": { 5 | "sourceMap": false, 6 | "types": ["cypress", "node"] 7 | } 8 | } 9 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "bracketSpacing": true, 3 | "tabWidth": 4, 4 | "singleQuote": true, 5 | "semi": true, 6 | "useTabs": false, 7 | "trailingComma": "es5", 8 | "printWidth": 140 9 | } 10 | -------------------------------------------------------------------------------- /.prettierignore: -------------------------------------------------------------------------------- 1 | package.json 2 | package-lock.json 3 | yarn.lock 4 | dist 5 | node_modules 6 | build 7 | res 8 | coverage 9 | libs/svg-management/src/lib/svg-canvas-editor/svg-canvas-editor.component.stories.ts 10 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/cypress/plugins/index.ts: -------------------------------------------------------------------------------- 1 | // Plugins enable you to tap into, modify, or extend the internal behavior of Cypress 2 | // For more info, visit https://on.cypress.io/plugins-api 3 | module.exports = (on, config) => {}; 4 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/ng-package.json: -------------------------------------------------------------------------------- 1 | { 2 | "$schema": "../../node_modules/ng-packagr/ng-package.schema.json", 3 | "dest": "../../dist/ngx-pinch-zoom", 4 | "lib": { 5 | "entryFile": "src/public-api.ts" 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/cypress/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "../../../tsconfig.json", 3 | "include": ["**/*.ts"], 4 | "compilerOptions": { 5 | "sourceMap": false, 6 | "types": ["cypress", "node"] 7 | } 8 | } 9 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/cypress/integration/spec.ts: -------------------------------------------------------------------------------- 1 | describe('My First Test', () => { 2 | it('Visits the initial project page', () => { 3 | cy.visit('/'); 4 | cy.contains('Welcome'); 5 | cy.contains('sandbox app is running!'); 6 | }); 7 | }); 8 | -------------------------------------------------------------------------------- /src/app/app-routing.module.ts: -------------------------------------------------------------------------------- 1 | import { NgModule } from '@angular/core'; 2 | import { RouterModule, Routes } from '@angular/router'; 3 | 4 | const routes: Routes = []; 5 | 6 | @NgModule({ 7 | imports: [RouterModule.forRoot(routes)], 8 | exports: [RouterModule], 9 | }) 10 | export class AppRoutingModule {} 11 | -------------------------------------------------------------------------------- /tsconfig.app.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "extends": "./tsconfig.json", 4 | "compilerOptions": { 5 | "outDir": "./out-tsc/app", 6 | "types": [] 7 | }, 8 | "files": ["src/main.ts", "src/polyfills.ts"], 9 | "include": ["src/**/*.d.ts"] 10 | } 11 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/tsconfig.lib.prod.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "extends": "./tsconfig.lib.json", 4 | "compilerOptions": { 5 | "declarationMap": false 6 | }, 7 | "angularCompilerOptions": { 8 | "compilationMode": "partial" 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | # Editor configuration, see https://editorconfig.org 2 | root = true 3 | 4 | [*] 5 | charset = utf-8 6 | indent_style = space 7 | indent_size = 4 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | 11 | [*.ts] 12 | quote_type = single 13 | 14 | [*.md] 15 | max_line_length = off 16 | trim_trailing_whitespace = false 17 | -------------------------------------------------------------------------------- /cypress.json: -------------------------------------------------------------------------------- 1 | { 2 | "integrationFolder": "cypress/integration", 3 | "supportFile": "cypress/support/index.ts", 4 | "videosFolder": "cypress/videos", 5 | "screenshotsFolder": "cypress/screenshots", 6 | "pluginsFile": "cypress/plugins/index.ts", 7 | "fixturesFolder": "cypress/fixtures", 8 | "baseUrl": "http://localhost:4200" 9 | } 10 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/tsconfig.spec.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "extends": "../../tsconfig.json", 4 | "compilerOptions": { 5 | "outDir": "../../out-tsc/spec", 6 | "types": ["jasmine"] 7 | }, 8 | "files": ["src/test.ts"], 9 | "include": ["**/*.spec.ts", "**/*.d.ts"] 10 | } 11 | -------------------------------------------------------------------------------- /tsconfig.spec.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "extends": "./tsconfig.json", 4 | "compilerOptions": { 5 | "outDir": "./out-tsc/spec", 6 | "types": ["jasmine"] 7 | }, 8 | "files": ["src/test.ts", "src/polyfills.ts"], 9 | "include": ["src/**/*.spec.ts", "src/**/*.d.ts"] 10 | } 11 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/lib/pinch-zoom.component.html: -------------------------------------------------------------------------------- 1 |
2 | 3 |
4 | 5 | 6 | @if (isControl()) { 7 |
12 | } 13 | -------------------------------------------------------------------------------- /src/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | IvypinchApp 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/app/app.component.ts: -------------------------------------------------------------------------------- 1 | import { Component } from '@angular/core'; 2 | 3 | @Component({ 4 | selector: 'app-root', 5 | templateUrl: './app.component.html', 6 | styleUrls: ['./app.component.sass'], 7 | standalone: false 8 | }) 9 | export class AppComponent { 10 | title = 'ivypinchApp'; 11 | public zoomstate = 1; 12 | 13 | onZoomChanged(zoom: number) { 14 | this.zoomstate = zoom; 15 | } 16 | } 17 | -------------------------------------------------------------------------------- /src/main.ts: -------------------------------------------------------------------------------- 1 | import { enableProdMode } from '@angular/core'; 2 | import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; 3 | 4 | import { AppModule } from './app/app.module'; 5 | import { environment } from './environments/environment'; 6 | 7 | if (environment.production) { 8 | enableProdMode(); 9 | } 10 | 11 | platformBrowserDynamic() 12 | .bootstrapModule(AppModule) 13 | .catch((err) => console.error(err)); 14 | -------------------------------------------------------------------------------- /src/test.ts: -------------------------------------------------------------------------------- 1 | // This file is required by karma.conf.js and loads recursively all the .spec and framework files 2 | 3 | import 'zone.js/dist/zone-testing'; 4 | import { getTestBed } from '@angular/core/testing'; 5 | import { BrowserDynamicTestingModule, platformBrowserDynamicTesting } from '@angular/platform-browser-dynamic/testing'; 6 | 7 | // First, initialize the Angular testing environment. 8 | getTestBed().initTestEnvironment(BrowserDynamicTestingModule, platformBrowserDynamicTesting()); 9 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/cypress.json: -------------------------------------------------------------------------------- 1 | { 2 | "integrationFolder": "projects/ngx-pinch-zoom/cypress/integration", 3 | "supportFile": "projects/ngx-pinch-zoom/cypress/support/index.ts", 4 | "videosFolder": "projects/ngx-pinch-zoom/cypress/videos", 5 | "screenshotsFolder": "projects/ngx-pinch-zoom/cypress/screenshots", 6 | "pluginsFile": "projects/ngx-pinch-zoom/cypress/plugins/index.ts", 7 | "fixturesFolder": "projects/ngx-pinch-zoom/cypress/fixtures", 8 | "baseUrl": "http://localhost:4200" 9 | } 10 | -------------------------------------------------------------------------------- /src/app/app.module.ts: -------------------------------------------------------------------------------- 1 | import { NgModule } from '@angular/core'; 2 | import { BrowserModule } from '@angular/platform-browser'; 3 | 4 | import { AppRoutingModule } from './app-routing.module'; 5 | import { AppComponent } from './app.component'; 6 | import {PinchZoomComponent} from "ngx-pinch-zoom"; 7 | 8 | @NgModule({ 9 | declarations: [AppComponent], 10 | imports: [BrowserModule, AppRoutingModule, PinchZoomComponent], 11 | providers: [], 12 | bootstrap: [AppComponent], 13 | }) 14 | export class AppModule {} 15 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/test.ts: -------------------------------------------------------------------------------- 1 | // This file is required by karma.conf.js and loads recursively all the .spec and framework files 2 | 3 | import 'zone.js/dist/zone'; 4 | import 'zone.js/dist/zone-testing'; 5 | import { getTestBed } from '@angular/core/testing'; 6 | import { BrowserDynamicTestingModule, platformBrowserDynamicTesting } from '@angular/platform-browser-dynamic/testing'; 7 | 8 | // First, initialize the Angular testing environment. 9 | getTestBed().initTestEnvironment(BrowserDynamicTestingModule, platformBrowserDynamicTesting()); 10 | -------------------------------------------------------------------------------- /cypress/integration/spec.ts: -------------------------------------------------------------------------------- 1 | Cypress.on('window:before:load', (win) => { 2 | cy.spy(win.console, 'error'); 3 | cy.spy(win.console, 'warn'); 4 | }); 5 | 6 | describe('workspace-project App', () => { 7 | afterEach(() => { 8 | cy.window().then((win) => { 9 | expect(win.console.error).to.have.callCount(0); 10 | expect(win.console.warn).to.have.callCount(0); 11 | }); 12 | }); 13 | 14 | it('should display welcome message', () => { 15 | cy.visit('/'); 16 | cy.get('title').should('contain.text', 'IvypinchApp'); 17 | }); 18 | }); 19 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/tsconfig.lib.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "extends": "../../tsconfig.json", 4 | "compilerOptions": { 5 | "outDir": "../../out-tsc/lib", 6 | "declaration": true, 7 | "declarationMap": true, 8 | "inlineSources": true, 9 | "types": [], 10 | "lib": ["dom", "es2018"] 11 | }, 12 | "angularCompilerOptions": { 13 | "skipTemplateCodegen": true, 14 | "strictMetadataEmit": true, 15 | "enableResourceInlining": true 16 | }, 17 | "exclude": ["src/test.ts", "**/*.spec.ts"] 18 | } 19 | -------------------------------------------------------------------------------- /cypress/support/index.ts: -------------------------------------------------------------------------------- 1 | // *********************************************************** 2 | // This example support/index.js is processed and 3 | // loaded automatically before your test files. 4 | // 5 | // This is a great place to put global configuration and 6 | // behavior that modifies Cypress. 7 | // 8 | // You can change the location of this file or turn off 9 | // automatically serving support files with the 10 | // 'supportFile' configuration option. 11 | // 12 | // You can read more here: 13 | // https://on.cypress.io/configuration 14 | // *********************************************************** 15 | 16 | // When a command from ./commands is ready to use, import with `import './commands'` syntax 17 | // import './commands'; 18 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/lib/interfaces.ts: -------------------------------------------------------------------------------- 1 | export interface Properties { 2 | element?: HTMLElement; 3 | doubleTap?: boolean; 4 | doubleTapScale?: number; 5 | zoomControlScale?: number; 6 | transitionDuration?: number; 7 | autoZoomOut?: boolean; 8 | limitZoom?: number | string | 'original image size'; 9 | disablePan?: boolean; 10 | limitPan?: boolean; 11 | minPanScale?: number; 12 | minScale?: number; 13 | listeners?: 'auto' | 'mouse and touch'; 14 | wheel?: boolean; 15 | fullImage?: { 16 | path: string; 17 | minScale?: number; 18 | }; 19 | autoHeight?: boolean; 20 | wheelZoomFactor?: number; 21 | draggableImage?: boolean; 22 | draggableOnPinch?: boolean; 23 | } 24 | -------------------------------------------------------------------------------- /src/environments/environment.ts: -------------------------------------------------------------------------------- 1 | // This file can be replaced during build by using the `fileReplacements` array. 2 | // `ng build --prod` replaces `environment.ts` with `environment.prod.ts`. 3 | // The list of file replacements can be found in `angular.json`. 4 | 5 | export const environment = { 6 | production: false, 7 | }; 8 | 9 | /* 10 | * For easier debugging in development mode, you can import the following file 11 | * to ignore zone related error stack frames such as `zone.run`, `zoneDelegate.invokeTask`. 12 | * 13 | * This import should be commented out in production mode because it will have a negative impact 14 | * on performance if an error is thrown. 15 | */ 16 | // import 'zone.js/dist/zone-error'; // Included with Angular CLI. 17 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/cypress/support/index.ts: -------------------------------------------------------------------------------- 1 | // *********************************************************** 2 | // This example support/index.js is processed and 3 | // loaded automatically before your test files. 4 | // 5 | // This is a great place to put global configuration and 6 | // behavior that modifies Cypress. 7 | // 8 | // You can change the location of this file or turn off 9 | // automatically serving support files with the 10 | // 'supportFile' configuration option. 11 | // 12 | // You can read more here: 13 | // https://on.cypress.io/configuration 14 | // *********************************************************** 15 | 16 | // When a command from ./commands is ready to use, import with `import './commands'` syntax 17 | // import './commands'; 18 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See http://help.github.com/ignore-files/ for more about ignoring files. 2 | 3 | # compiled output 4 | /dist 5 | /tmp 6 | /out-tsc 7 | # Only exists if Bazel was run 8 | /bazel-out 9 | 10 | # dependencies 11 | /node_modules 12 | 13 | # profiling files 14 | chrome-profiler-events*.json 15 | speed-measure-plugin*.json 16 | 17 | # IDEs and editors 18 | /.idea 19 | .project 20 | .classpath 21 | .c9/ 22 | *.launch 23 | .settings/ 24 | *.sublime-workspace 25 | 26 | # IDE - VSCode 27 | .vscode/* 28 | !.vscode/settings.json 29 | !.vscode/tasks.json 30 | !.vscode/launch.json 31 | !.vscode/extensions.json 32 | .history/* 33 | 34 | # misc 35 | /.angular/cache 36 | /.sass-cache 37 | /connect.lock 38 | /coverage 39 | /libpeerconnection.log 40 | npm-debug.log 41 | yarn-error.log 42 | testem.log 43 | /typings 44 | 45 | # System Files 46 | .DS_Store 47 | Thumbs.db 48 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | /* To learn more about this file see: https://angular.io/config/tsconfig. */ 2 | { 3 | "compileOnSave": false, 4 | "compilerOptions": { 5 | "baseUrl": "./", 6 | "outDir": "./dist/out-tsc", 7 | "sourceMap": true, 8 | "esModuleInterop": true, 9 | "declaration": false, 10 | "experimentalDecorators": true, 11 | "moduleResolution": "bundler", 12 | "importHelpers": true, 13 | "paths": { 14 | "ngx-pinch-zoom": [ 15 | "dist/ngx-pinch-zoom/ngx-pinch-zoom", 16 | "dist/ngx-pinch-zoom" 17 | ] 18 | }, 19 | "target": "ES2022", 20 | "module": "es2020", 21 | "lib": [ 22 | "es2018", 23 | "dom" 24 | ], 25 | "types": [ 26 | "node" 27 | ], 28 | "useDefineForClassFields": false 29 | }, 30 | "angularCompilerOptions": { 31 | "enableI18nLegacyMessageIdFormat": false 32 | } 33 | } 34 | -------------------------------------------------------------------------------- /src/app/app.component.spec.ts: -------------------------------------------------------------------------------- 1 | import { TestBed } from '@angular/core/testing'; 2 | import { RouterTestingModule } from '@angular/router/testing'; 3 | import { AppComponent } from './app.component'; 4 | import { PinchZoomModule } from '../../projects/ngx-pinch-zoom/src/lib/pinch-zoom.module'; 5 | 6 | describe('AppComponent', () => { 7 | beforeEach(async () => { 8 | await TestBed.configureTestingModule({ 9 | imports: [RouterTestingModule, PinchZoomModule], 10 | declarations: [AppComponent], 11 | }).compileComponents(); 12 | }); 13 | 14 | it('should create the app', () => { 15 | const fixture = TestBed.createComponent(AppComponent); 16 | const app = fixture.componentInstance; 17 | expect(app).toBeTruthy(); 18 | }); 19 | 20 | it(`should have as title 'ivypinchApp'`, () => { 21 | const fixture = TestBed.createComponent(AppComponent); 22 | const app = fixture.componentInstance; 23 | expect(app.title).toEqual('ivypinchApp'); 24 | }); 25 | }); 26 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/lib/properties.ts: -------------------------------------------------------------------------------- 1 | import { Properties } from './interfaces'; 2 | 3 | export const defaultProperties: Properties = { 4 | transitionDuration: 200, 5 | doubleTap: true, 6 | doubleTapScale: 2, 7 | limitZoom: 'original image size', 8 | autoZoomOut: false, 9 | zoomControlScale: 1, 10 | minPanScale: 1.0001, 11 | minScale: 0, 12 | listeners: 'mouse and touch', 13 | wheel: true, 14 | wheelZoomFactor: 0.2, 15 | draggableImage: false, 16 | draggableOnPinch: false, 17 | }; 18 | 19 | export const backwardCompatibilityProperties = { 20 | 'transition-duration': 'transitionDuration', 21 | transitionDurationBackwardCompatibility: 'transitionDuration', 22 | 'double-tap': 'doubleTap', 23 | doubleTapBackwardCompatibility: 'doubleTap', 24 | 'double-tap-scale': 'doubleTapScale', 25 | doubleTapScaleBackwardCompatibility: 'doubleTapScale', 26 | 'auto-zoom-out': 'autoZoomOut', 27 | autoZoomOutBackwardCompatibility: 'autoZoomOut', 28 | 'limit-zoom': 'limitZoom', 29 | limitZoomBackwardCompatibility: 'limitZoom', 30 | }; 31 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2019 Nikita Drozhzhin 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@meddv/ngx-pinch-zoom", 3 | "version": "20.0.1", 4 | "description": "Pinch zoom component for Angular 20.", 5 | "repository": { 6 | "type": "git", 7 | "url": "git+https://github.com/medDV-GmbH/ngx-pinch-zoom" 8 | }, 9 | "author": { 10 | "name": "Nikita Drozhzhin", 11 | "email": "drozhzhin.n.e@gmail.com" 12 | }, 13 | "contributors": [ 14 | { 15 | "name": "Konstantin Schütte", 16 | "email": "team-web@meddv.de", 17 | "url": "https://www.meddv.de" 18 | }, 19 | { 20 | "name": "Björn Schmidt", 21 | "email": "team-web@meddv.de", 22 | "url": "https://www.meddv.de" 23 | } 24 | ], 25 | "homepage": "http://www.meddv.de", 26 | "keywords": [ 27 | "Angular", 28 | "Angular 20", 29 | "ngx", 30 | "Pinch zoom", 31 | "Image zoom", 32 | "Touch image zoom" 33 | ], 34 | "license": "MIT", 35 | "bugs": { 36 | "url": "https://github.com/medDV-GmbH/ngx-pinch-zoom/issues" 37 | }, 38 | "peerDependencies": { 39 | "@angular/common": ">=20.0.0", 40 | "@angular/core": ">=20.0.0" 41 | }, 42 | "dependencies": { 43 | "tslib": "^2.0.0" 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /cypress/support/commands.ts: -------------------------------------------------------------------------------- 1 | // *********************************************** 2 | // This example namespace declaration will help 3 | // with Intellisense and code completion in your 4 | // IDE or Text Editor. 5 | // *********************************************** 6 | // declare namespace Cypress { 7 | // interface Chainable { 8 | // customCommand(param: any): typeof customCommand; 9 | // } 10 | // } 11 | // 12 | // function customCommand(param: any): void { 13 | // console.warn(param); 14 | // } 15 | // 16 | // NOTE: You can use it like so: 17 | // Cypress.Commands.add('customCommand', customCommand); 18 | // 19 | // *********************************************** 20 | // This example commands.js shows you how to 21 | // create various custom commands and overwrite 22 | // existing commands. 23 | // 24 | // For more comprehensive examples of custom 25 | // commands please read more here: 26 | // https://on.cypress.io/custom-commands 27 | // *********************************************** 28 | // 29 | // 30 | // -- This is a parent command -- 31 | // Cypress.Commands.add("login", (email, password) => { ... }) 32 | // 33 | // 34 | // -- This is a child command -- 35 | // Cypress.Commands.add("drag", { prevSubject: 'element'}, (subject, options) => { ... }) 36 | // 37 | // 38 | // -- This is a dual command -- 39 | // Cypress.Commands.add("dismiss", { prevSubject: 'optional'}, (subject, options) => { ... }) 40 | // 41 | // 42 | // -- This will overwrite an existing command -- 43 | // Cypress.Commands.overwrite("visit", (originalFn, url, options) => { ... }) 44 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/cypress/support/commands.ts: -------------------------------------------------------------------------------- 1 | // *********************************************** 2 | // This example namespace declaration will help 3 | // with Intellisense and code completion in your 4 | // IDE or Text Editor. 5 | // *********************************************** 6 | // declare namespace Cypress { 7 | // interface Chainable { 8 | // customCommand(param: any): typeof customCommand; 9 | // } 10 | // } 11 | // 12 | // function customCommand(param: any): void { 13 | // console.warn(param); 14 | // } 15 | // 16 | // NOTE: You can use it like so: 17 | // Cypress.Commands.add('customCommand', customCommand); 18 | // 19 | // *********************************************** 20 | // This example commands.js shows you how to 21 | // create various custom commands and overwrite 22 | // existing commands. 23 | // 24 | // For more comprehensive examples of custom 25 | // commands please read more here: 26 | // https://on.cypress.io/custom-commands 27 | // *********************************************** 28 | // 29 | // 30 | // -- This is a parent command -- 31 | // Cypress.Commands.add("login", (email, password) => { ... }) 32 | // 33 | // 34 | // -- This is a child command -- 35 | // Cypress.Commands.add("drag", { prevSubject: 'element'}, (subject, options) => { ... }) 36 | // 37 | // 38 | // -- This is a dual command -- 39 | // Cypress.Commands.add("dismiss", { prevSubject: 'optional'}, (subject, options) => { ... }) 40 | // 41 | // 42 | // -- This will overwrite an existing command -- 43 | // Cypress.Commands.overwrite("visit", (originalFn, url, options) => { ... }) 44 | -------------------------------------------------------------------------------- /karma.conf.js: -------------------------------------------------------------------------------- 1 | // Karma configuration file, see link for more information 2 | // https://karma-runner.github.io/1.0/config/configuration-file.html 3 | 4 | module.exports = function (config) { 5 | config.set({ 6 | basePath: '', 7 | frameworks: ['jasmine', '@angular-devkit/build-angular'], 8 | plugins: [ 9 | require('karma-jasmine'), 10 | require('karma-chrome-launcher'), 11 | require('karma-jasmine-html-reporter'), 12 | require('karma-coverage'), 13 | require('@angular-devkit/build-angular/plugins/karma'), 14 | ], 15 | client: { 16 | jasmine: { 17 | // you can add configuration options for Jasmine here 18 | // the possible options are listed at https://jasmine.github.io/api/edge/Configuration.html 19 | // for example, you can disable the random execution with `random: false` 20 | // or set a specific seed with `seed: 4321` 21 | }, 22 | clearContext: false, // leave Jasmine Spec Runner output visible in browser 23 | }, 24 | jasmineHtmlReporter: { 25 | suppressAll: true, // removes the duplicated traces 26 | }, 27 | coverageReporter: { 28 | dir: require('path').join(__dirname, './coverage/ivypinchApp'), 29 | subdir: '.', 30 | reporters: [{ type: 'html' }, { type: 'text-summary' }], 31 | }, 32 | reporters: ['progress', 'kjhtml'], 33 | port: 9876, 34 | colors: true, 35 | logLevel: config.LOG_INFO, 36 | autoWatch: true, 37 | browsers: ['Chrome'], 38 | singleRun: false, 39 | restartOnFileChange: true, 40 | }); 41 | }; 42 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/karma.conf.js: -------------------------------------------------------------------------------- 1 | // Karma configuration file, see link for more information 2 | // https://karma-runner.github.io/1.0/config/configuration-file.html 3 | 4 | module.exports = function (config) { 5 | config.set({ 6 | basePath: '', 7 | frameworks: ['jasmine', '@angular-devkit/build-angular'], 8 | plugins: [ 9 | require('karma-jasmine'), 10 | require('karma-chrome-launcher'), 11 | require('karma-jasmine-html-reporter'), 12 | require('karma-coverage'), 13 | require('@angular-devkit/build-angular/plugins/karma'), 14 | ], 15 | client: { 16 | jasmine: { 17 | // you can add configuration options for Jasmine here 18 | // the possible options are listed at https://jasmine.github.io/api/edge/Configuration.html 19 | // for example, you can disable the random execution with `random: false` 20 | // or set a specific seed with `seed: 4321` 21 | }, 22 | clearContext: false, // leave Jasmine Spec Runner output visible in browser 23 | }, 24 | jasmineHtmlReporter: { 25 | suppressAll: true, // removes the duplicated traces 26 | }, 27 | coverageReporter: { 28 | dir: require('path').join(__dirname, '../../coverage/ngx-pinch-zoom'), 29 | subdir: '.', 30 | reporters: [{ type: 'html' }, { type: 'text-summary' }], 31 | }, 32 | reporters: ['progress', 'kjhtml'], 33 | port: 9876, 34 | colors: true, 35 | logLevel: config.LOG_INFO, 36 | autoWatch: true, 37 | browsers: ['Chrome'], 38 | singleRun: false, 39 | restartOnFileChange: true, 40 | }); 41 | }; 42 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ngx-pinch-zoom", 3 | "version": "20.0.0", 4 | "scripts": { 5 | "ng": "ng", 6 | "start": "ng serve", 7 | "build": "ng build", 8 | "build-lib": "npx ng-packagr -p projects/ngx-pinch-zoom/ng-package.json", 9 | "test": "ng test", 10 | "lint": "ng lint", 11 | "e2e": "ng e2e", 12 | "cypress:open": "cypress open", 13 | "cypress:run": "cypress run" 14 | }, 15 | "dependencies": { 16 | "@angular/animations": "^20.3.15", 17 | "@angular/common": "^20.3.15", 18 | "@angular/compiler": "^20.3.15", 19 | "@angular/core": "^20.3.15", 20 | "@angular/forms": "^20.3.15", 21 | "@angular/platform-browser": "^20.3.15", 22 | "@angular/platform-browser-dynamic": "^20.3.15", 23 | "@angular/router": "^20.3.15", 24 | "rxjs": "~7.5.2", 25 | "tslib": "^2.3.1", 26 | "zone.js": "0.15.0" 27 | }, 28 | "devDependencies": { 29 | "@angular-devkit/build-angular": "^20.3.12", 30 | "@angular-eslint/eslint-plugin": "^20.7.0", 31 | "@angular-eslint/eslint-plugin-template": "^20.7.0", 32 | "@angular/cli": "^20.3.12", 33 | "@angular/compiler-cli": "^20.3.15", 34 | "@cypress/schematic": "^1.6.0", 35 | "@types/jasmine": "~3.10.3", 36 | "@types/node": "^16.18.24", 37 | "@typescript-eslint/eslint-plugin": "^5.10.2", 38 | "@typescript-eslint/parser": "^5.10.2", 39 | "cypress": "latest", 40 | "eslint": "^8.8.0", 41 | "eslint-plugin-import": "^2.25.4", 42 | "eslint-plugin-jsdoc": "^37.7.1", 43 | "eslint-plugin-prefer-arrow": "^1.2.3", 44 | "jasmine-core": "~4.6.1", 45 | "jasmine-spec-reporter": "~7.0.0", 46 | "karma": "~6.4.4", 47 | "karma-chrome-launcher": "~3.2.0", 48 | "karma-coverage": "~2.2.1", 49 | "karma-jasmine": "~5.1.0", 50 | "karma-jasmine-html-reporter": "^2.1.0", 51 | "ng-packagr": "^20.3.2", 52 | "prettier": "^2.7.1", 53 | "ts-node": "~10.4.0", 54 | "typescript": "5.9.3" 55 | } 56 | } 57 | -------------------------------------------------------------------------------- /src/polyfills.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * This file includes polyfills needed by Angular and is loaded before the app. 3 | * You can add your own extra polyfills to this file. 4 | * 5 | * This file is divided into 2 sections: 6 | * 1. Browser polyfills. These are applied before loading ZoneJS and are sorted by browsers. 7 | * 2. Application imports. Files imported after ZoneJS that should be loaded before your main 8 | * file. 9 | * 10 | * The current setup is for so-called "evergreen" browsers; the last versions of browsers that 11 | * automatically update themselves. This includes Safari >= 10, Chrome >= 55 (including Opera), 12 | * Edge >= 13 on the desktop, and iOS 10 and Chrome on mobile. 13 | * 14 | * Learn more in https://angular.io/guide/browser-support 15 | */ 16 | 17 | /*************************************************************************************************** 18 | * BROWSER POLYFILLS 19 | */ 20 | 21 | /** 22 | * IE11 requires the following for NgClass support on SVG elements 23 | */ 24 | // import 'classlist.js'; // Run `npm install --save classlist.js`. 25 | 26 | /** 27 | * Web Animations `@angular/platform-browser/animations` 28 | * Only required if AnimationBuilder is used within the application and using IE/Edge or Safari. 29 | * Standard animation support in Angular DOES NOT require any polyfills (as of Angular 6.0). 30 | */ 31 | // import 'web-animations-js'; // Run `npm install --save web-animations-js`. 32 | 33 | /** 34 | * By default, zone.js will patch all possible macroTask and DomEvents 35 | * user can disable parts of macroTask/DomEvents patch by setting following flags 36 | * because those flags need to be set before `zone.js` being loaded, and webpack 37 | * will put import in the top of bundle, so user need to create a separate file 38 | * in this directory (for example: zone-flags.ts), and put the following flags 39 | * into that file, and then add the following code before importing zone.js. 40 | * import './zone-flags'; 41 | * 42 | * The flags allowed in zone-flags.ts are listed here. 43 | * 44 | * The following flags will work for all browsers. 45 | * 46 | * (window as any).__Zone_disable_requestAnimationFrame = true; // disable patch requestAnimationFrame 47 | * (window as any).__Zone_disable_on_property = true; // disable patch onProperty such as onclick 48 | * (window as any).__zone_symbol__UNPATCHED_EVENTS = ['scroll', 'mousemove']; // disable patch specified eventNames 49 | * 50 | * in IE/Edge developer tools, the addEventListener will also be wrapped by zone.js 51 | * with the following flag, it will bypass `zone.js` patch for IE/Edge 52 | * 53 | * (window as any).__Zone_enable_cross_context_check = true; 54 | * 55 | */ 56 | 57 | /*************************************************************************************************** 58 | * Zone JS is required by default for Angular itself. 59 | */ 60 | import 'zone.js'; // Included with Angular CLI. 61 | 62 | /*************************************************************************************************** 63 | * APPLICATION IMPORTS 64 | */ 65 | -------------------------------------------------------------------------------- /src/app/app.component.sass: -------------------------------------------------------------------------------- 1 | .wrapper 2 | max-width: 500px 3 | margin: 0 auto 4 | font-family: system-ui 5 | 6 | .demo 7 | margin-bottom: 60px 8 | 9 | .actions 10 | padding-top: 20px 11 | 12 | h1 13 | font-size: 30px 14 | h2 15 | font-size: 20px 16 | 17 | /* Colors */ 18 | 19 | $black: #00000a 20 | $white: #fff 21 | $dark-white: rgba(0, 0, 0, 0.75) 22 | $blue: #0366d6 23 | $dark-blue: #035cbf 24 | $green: #7cae7a 25 | $dark-green: #709a6e 26 | $gray: rgba(0, 0, 0, 0.5) 27 | $light-shadow: rgba(0, 0, 0, 0.15) 28 | 29 | /* Buttons */ 30 | 31 | .btn 32 | display: inline-block 33 | min-width: 190px 34 | line-height: 56px 35 | font-weight: 500 36 | border-radius: 6px 37 | text-align: center 38 | margin: 0 15px 15px 0 39 | background: none 40 | transition: all 200ms 41 | color: $black 42 | outline: none 43 | font-size: 15px 44 | letter-spacing: 1px 45 | text-transform: uppercase 46 | font-weight: 600 47 | background: $white 48 | .btn:hover 49 | color: $dark-white 50 | text-decoration: none 51 | cursor: pointer 52 | box-shadow: 0px 3px 10px $light-shadow 53 | .btn:active 54 | position: relative 55 | top: 2px 56 | 57 | .btn-small 58 | line-height: 32px 59 | font-weight: normal 60 | padding: 0 8px 61 | font-size: 14px 62 | letter-spacing: 0px 63 | min-width: auto 64 | text-transform: none 65 | 66 | .btn-outline 67 | border: 1px solid #999 68 | .btn-outline:hover 69 | border-color: $dark-white 70 | .btn-rounded 71 | border-radius: 100px 72 | .btn[disabled] 73 | cursor: default 74 | box-shadow: none 75 | 76 | .btn-light 77 | color: $white 78 | .btn-light:hover 79 | color: rgba(255, 255, 255, 0.75) 80 | .btn-light.btn-outline 81 | border: 1px solid $white 82 | .btn-light.btn-outline:hover 83 | border-color: rgba(255, 255, 255, 0.75) 84 | 85 | .btn-green 86 | background: $green 87 | color: $white 88 | border: none 89 | .btn-green.btn-outline 90 | border: 1px solid $green 91 | color: $green 92 | background: none 93 | .btn-green.btn-clear 94 | color: $green 95 | .btn-green:hover 96 | background: $dark-green 97 | color: $white 98 | .btn-green.btn-outline:hover 99 | background: none 100 | border-color: $dark-green 101 | color: $dark-green 102 | .btn-green.btn-clear:hover 103 | color: $dark-green 104 | .btn-green[disabled] 105 | background: #aaa 106 | .btn-green.btn-outline[disabled] 107 | background: none 108 | border-color: #aaa 109 | color: #aaa 110 | 111 | .btn-blue 112 | background: $blue 113 | color: $white 114 | border: none 115 | .btn-blue.btn-outline 116 | border: 1px solid $blue 117 | color: $blue 118 | background: none 119 | .btn-blue.btn-clear 120 | color: $blue 121 | background: none 122 | .btn-blue:hover 123 | background: $dark-blue 124 | color: $white 125 | .btn-blue.btn-outline:hover 126 | background: none 127 | border-color: $dark-blue 128 | color: $dark-blue 129 | .btn-blue.btn-clear:hover 130 | color: $dark-blue 131 | 132 | .btn-clear 133 | background: none 134 | border: none 135 | .btn-clear:hover 136 | background: none -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/lib/pinch-zoom.component.sass: -------------------------------------------------------------------------------- 1 | :host 2 | position: relative 3 | overflow: hidden 4 | display: block 5 | 6 | .pinch-zoom-content 7 | height: inherit 8 | 9 | .pz-dragging 10 | cursor: all-scroll 11 | 12 | 13 | /* Zoom button */ 14 | .pz-zoom-button 15 | position: absolute 16 | z-index: 1000 17 | color: #fff 18 | background-image: url(), url() 19 | background-color: rgba(0, 0, 0, .8) 20 | background-position: center, -1000px 21 | background-repeat: no-repeat, no-repeat 22 | background-size: 40px 23 | width: 56px 24 | height: 56px 25 | border-radius: 4px 26 | opacity: 0.5 27 | cursor: pointer 28 | transition: opacity .1s 29 | user-select: none 30 | 31 | .pz-zoom-button-out 32 | background-position: -1000px, center 33 | 34 | .pz-zoom-button:hover 35 | opacity: 0.7 36 | 37 | .pz-zoom-button.pz-zoom-control-position-right 38 | right: 16px 39 | top: 50% 40 | margin-top: -28px 41 | 42 | .pz-zoom-button.pz-zoom-control-position-right-bottom 43 | right: 16px 44 | bottom: 32px 45 | 46 | .pz-zoom-button.pz-zoom-control-position-bottom 47 | bottom: 16px 48 | left: 50% 49 | margin-left: -28px 50 | 51 | 52 | /* Zoom control */ 53 | .pz-zoom-control 54 | position: absolute 55 | background-color: rgba(0, 0, 0, .8) 56 | border-radius: 4px 57 | overflow: hidden 58 | 59 | .pz-zoom-control.pz-zoom-control-position-right 60 | right: 16px 61 | top: 50% 62 | margin-top: -48px 63 | 64 | .pz-zoom-control.pz-zoom-control-position-right-bottom 65 | right: 16px 66 | bottom: 32px 67 | 68 | .pz-zoom-control.pz-zoom-control-position-bottom 69 | bottom: 16px 70 | left: 50% 71 | margin-left: -48px 72 | 73 | .pz-zoom-in, 74 | .pz-zoom-out 75 | width: 48px 76 | height: 48px 77 | background-position: center 78 | background-repeat: no-repeat 79 | opacity: 1 80 | cursor: pointer 81 | 82 | .pz-zoom-in:hover, 83 | .pz-zoom-out:hover 84 | background-color: rgba(255, 255, 255, 0.2) 85 | 86 | .pz-zoom-control-position-bottom .pz-zoom-in, 87 | .pz-zoom-control-position-bottom .pz-zoom-out 88 | float: right 89 | 90 | .pz-disabled 91 | opacity: 0.5 92 | cursor: default 93 | 94 | .pz-disabled:hover 95 | background-color: rgba(255, 255, 255, 0) 96 | 97 | .pz-zoom-in 98 | background-image: url() 99 | 100 | .pz-zoom-out 101 | background-image: url() 102 | -------------------------------------------------------------------------------- /angular.json: -------------------------------------------------------------------------------- 1 | { 2 | "$schema": "./node_modules/@angular/cli/lib/config/schema.json", 3 | "cli": { 4 | "analytics": false 5 | }, 6 | "version": 1, 7 | "newProjectRoot": "projects", 8 | "projects": { 9 | "ivypinchApp": { 10 | "projectType": "application", 11 | "schematics": { 12 | "@schematics/angular:component": { 13 | "style": "sass" 14 | } 15 | }, 16 | "root": "", 17 | "sourceRoot": "src", 18 | "prefix": "app", 19 | "architect": { 20 | "build": { 21 | "builder": "@angular-devkit/build-angular:application", 22 | "options": { 23 | "outputPath": { 24 | "base": "dist/ivypinchApp" 25 | }, 26 | "index": "src/index.html", 27 | "polyfills": [ 28 | "src/polyfills.ts" 29 | ], 30 | "tsConfig": "tsconfig.app.json", 31 | "aot": true, 32 | "assets": ["src/favicon.ico", "src/assets"], 33 | "styles": ["src/styles.sass"], 34 | "scripts": [], 35 | "browser": "src/main.ts" 36 | }, 37 | "configurations": { 38 | "production": { 39 | "fileReplacements": [ 40 | { 41 | "replace": "src/environments/environment.ts", 42 | "with": "src/environments/environment.prod.ts" 43 | } 44 | ], 45 | "optimization": true, 46 | "outputHashing": "all", 47 | "sourceMap": false, 48 | "namedChunks": false, 49 | "extractLicenses": true, 50 | "budgets": [ 51 | { 52 | "type": "initial", 53 | "maximumWarning": "2mb", 54 | "maximumError": "5mb" 55 | }, 56 | { 57 | "type": "anyComponentStyle", 58 | "maximumWarning": "6kb", 59 | "maximumError": "10kb" 60 | } 61 | ] 62 | } 63 | } 64 | }, 65 | "serve": { 66 | "builder": "@angular-devkit/build-angular:dev-server", 67 | "options": { 68 | "buildTarget": "ivypinchApp:build" 69 | }, 70 | "configurations": { 71 | "production": { 72 | "buildTarget": "ivypinchApp:build:production" 73 | } 74 | } 75 | }, 76 | "extract-i18n": { 77 | "builder": "@angular-devkit/build-angular:extract-i18n", 78 | "options": { 79 | "buildTarget": "ivypinchApp:build" 80 | } 81 | }, 82 | "test": { 83 | "builder": "@angular-devkit/build-angular:karma", 84 | "options": { 85 | "main": "src/test.ts", 86 | "polyfills": "src/polyfills.ts", 87 | "tsConfig": "tsconfig.spec.json", 88 | "karmaConfig": "karma.conf.js", 89 | "assets": ["src/favicon.ico", "src/assets"], 90 | "styles": ["src/styles.sass"], 91 | "scripts": [] 92 | } 93 | }, 94 | "lint": { 95 | "builder": "@angular-devkit/build-angular:tslint", 96 | "options": { 97 | "tsConfig": ["tsconfig.app.json", "tsconfig.spec.json", "cypress/tsconfig.json"], 98 | "exclude": ["**/node_modules/**"] 99 | } 100 | }, 101 | "e2e": { 102 | "builder": "@cypress/schematic:cypress", 103 | "options": { 104 | "devServerTarget": "ivypinchApp:serve", 105 | "watch": true, 106 | "headless": false 107 | }, 108 | "configurations": { 109 | "production": { 110 | "devServerTarget": "ivypinchApp:serve:production" 111 | } 112 | } 113 | }, 114 | "cypress-run": { 115 | "builder": "@cypress/schematic:cypress", 116 | "options": { 117 | "devServerTarget": "ivypinchApp:serve" 118 | }, 119 | "configurations": { 120 | "production": { 121 | "devServerTarget": "ivypinchApp:serve:production" 122 | } 123 | } 124 | }, 125 | "cypress-open": { 126 | "builder": "@cypress/schematic:cypress", 127 | "options": { 128 | "watch": true, 129 | "headless": false 130 | } 131 | } 132 | } 133 | }, 134 | "ngx-pinch-zoom": { 135 | "projectType": "library", 136 | "root": "projects/ngx-pinch-zoom", 137 | "sourceRoot": "projects/ngx-pinch-zoom/src", 138 | "prefix": "lib", 139 | "architect": { 140 | "build": { 141 | "builder": "@angular-devkit/build-angular:ng-packagr", 142 | "options": { 143 | "tsConfig": "projects/ngx-pinch-zoom/tsconfig.lib.json", 144 | "project": "projects/ngx-pinch-zoom/ng-package.json" 145 | }, 146 | "configurations": { 147 | "production": { 148 | "tsConfig": "projects/ngx-pinch-zoom/tsconfig.lib.prod.json" 149 | } 150 | } 151 | }, 152 | "test": { 153 | "builder": "@angular-devkit/build-angular:karma", 154 | "options": { 155 | "main": "projects/ngx-pinch-zoom/src/test.ts", 156 | "tsConfig": "projects/ngx-pinch-zoom/tsconfig.spec.json", 157 | "karmaConfig": "projects/ngx-pinch-zoom/karma.conf.js" 158 | } 159 | }, 160 | "lint": { 161 | "builder": "@angular-devkit/build-angular:tslint", 162 | "options": { 163 | "tsConfig": [ 164 | "projects/ngx-pinch-zoom/tsconfig.lib.json", 165 | "projects/ngx-pinch-zoom/tsconfig.spec.json", 166 | "projects/ngx-pinch-zoom/cypress/tsconfig.json" 167 | ], 168 | "exclude": ["**/node_modules/**"] 169 | } 170 | }, 171 | "cypress-run": { 172 | "builder": "@cypress/schematic:cypress", 173 | "options": { 174 | "devServerTarget": "ngx-pinch-zoom:serve", 175 | "configFile": "projects/ngx-pinch-zoom/cypress.json" 176 | }, 177 | "configurations": { 178 | "production": { 179 | "devServerTarget": "ngx-pinch-zoom:serve:production" 180 | } 181 | } 182 | }, 183 | "cypress-open": { 184 | "builder": "@cypress/schematic:cypress", 185 | "options": { 186 | "watch": true, 187 | "headless": false, 188 | "configFile": "projects/ngx-pinch-zoom/cypress.json" 189 | } 190 | }, 191 | "e2e": { 192 | "builder": "@cypress/schematic:cypress", 193 | "options": { 194 | "devServerTarget": "ngx-pinch-zoom:serve", 195 | "watch": true, 196 | "headless": false 197 | }, 198 | "configurations": { 199 | "production": { 200 | "devServerTarget": "ngx-pinch-zoom:serve:production" 201 | } 202 | } 203 | } 204 | } 205 | } 206 | }, 207 | "schematics": { 208 | "@schematics/angular:component": { 209 | "type": "component" 210 | }, 211 | "@schematics/angular:directive": { 212 | "type": "directive" 213 | }, 214 | "@schematics/angular:service": { 215 | "type": "service" 216 | }, 217 | "@schematics/angular:guard": { 218 | "typeSeparator": "." 219 | }, 220 | "@schematics/angular:interceptor": { 221 | "typeSeparator": "." 222 | }, 223 | "@schematics/angular:module": { 224 | "typeSeparator": "." 225 | }, 226 | "@schematics/angular:pipe": { 227 | "typeSeparator": "." 228 | }, 229 | "@schematics/angular:resolver": { 230 | "typeSeparator": "." 231 | } 232 | } 233 | } 234 | -------------------------------------------------------------------------------- /.eslintrc.js: -------------------------------------------------------------------------------- 1 | /* 2 | 👋 Hi! This file was autogenerated by tslint-to-eslint-config. 3 | https://github.com/typescript-eslint/tslint-to-eslint-config 4 | 5 | It represents the closest reasonable ESLint configuration to this 6 | project's original TSLint configuration. 7 | 8 | We recommend eventually switching this configuration to extend from 9 | the recommended rulesets in typescript-eslint. 10 | https://github.com/typescript-eslint/tslint-to-eslint-config/blob/master/docs/FAQs.md 11 | 12 | Happy linting! 💖 13 | */ 14 | module.exports = { 15 | env: { 16 | browser: true, 17 | es6: true, 18 | node: true, 19 | }, 20 | extends: ['plugin:@typescript-eslint/recommended', 'plugin:@typescript-eslint/recommended-requiring-type-checking'], 21 | parser: '@typescript-eslint/parser', 22 | parserOptions: { 23 | project: 'tsconfig.json', 24 | sourceType: 'module', 25 | }, 26 | plugins: [ 27 | 'eslint-plugin-import', 28 | 'eslint-plugin-jsdoc', 29 | '@angular-eslint/eslint-plugin', 30 | '@angular-eslint/eslint-plugin-template', 31 | 'eslint-plugin-prefer-arrow', 32 | '@typescript-eslint', 33 | '@typescript-eslint/tslint', 34 | ], 35 | rules: { 36 | '@angular-eslint/component-class-suffix': 'error', 37 | '@angular-eslint/component-selector': [ 38 | 'error', 39 | { 40 | type: 'element', 41 | prefix: 'app', 42 | style: 'kebab-case', 43 | }, 44 | ], 45 | '@angular-eslint/contextual-lifecycle': 'error', 46 | '@angular-eslint/directive-class-suffix': 'error', 47 | '@angular-eslint/directive-selector': [ 48 | 'error', 49 | { 50 | type: 'attribute', 51 | prefix: 'app', 52 | style: 'camelCase', 53 | }, 54 | ], 55 | '@angular-eslint/no-conflicting-lifecycle': 'error', 56 | '@angular-eslint/no-host-metadata-property': 'error', 57 | '@angular-eslint/no-input-rename': 'error', 58 | '@angular-eslint/no-inputs-metadata-property': 'error', 59 | '@angular-eslint/no-output-native': 'error', 60 | '@angular-eslint/no-output-on-prefix': 'error', 61 | '@angular-eslint/no-output-rename': 'error', 62 | '@angular-eslint/no-outputs-metadata-property': 'error', 63 | '@angular-eslint/template/banana-in-box': 'error', 64 | '@angular-eslint/template/eqeqeq': 'error', 65 | '@angular-eslint/template/no-negated-async': 'error', 66 | '@angular-eslint/use-lifecycle-interface': 'error', 67 | '@angular-eslint/use-pipe-transform-interface': 'error', 68 | '@typescript-eslint/adjacent-overload-signatures': 'error', 69 | '@typescript-eslint/array-type': 'off', 70 | '@typescript-eslint/ban-types': [ 71 | 'error', 72 | { 73 | types: { 74 | Object: { 75 | message: 'Avoid using the `Object` type. Did you mean `object`?', 76 | }, 77 | Function: { 78 | message: 'Avoid using the `Function` type. Prefer a specific function type, like `() => void`.', 79 | }, 80 | Boolean: { 81 | message: 'Avoid using the `Boolean` type. Did you mean `boolean`?', 82 | }, 83 | Number: { 84 | message: 'Avoid using the `Number` type. Did you mean `number`?', 85 | }, 86 | String: { 87 | message: 'Avoid using the `String` type. Did you mean `string`?', 88 | }, 89 | Symbol: { 90 | message: 'Avoid using the `Symbol` type. Did you mean `symbol`?', 91 | }, 92 | }, 93 | }, 94 | ], 95 | '@typescript-eslint/consistent-type-assertions': 'error', 96 | '@typescript-eslint/dot-notation': 'error', 97 | '@typescript-eslint/indent': [ 98 | 'error', 99 | 4, 100 | { 101 | FunctionDeclaration: { 102 | parameters: 'first', 103 | }, 104 | FunctionExpression: { 105 | parameters: 'first', 106 | }, 107 | }, 108 | ], 109 | '@typescript-eslint/member-delimiter-style': [ 110 | 'error', 111 | { 112 | multiline: { 113 | delimiter: 'semi', 114 | requireLast: true, 115 | }, 116 | singleline: { 117 | delimiter: 'semi', 118 | requireLast: false, 119 | }, 120 | }, 121 | ], 122 | '@typescript-eslint/member-ordering': 'error', 123 | '@typescript-eslint/naming-convention': 'error', 124 | '@typescript-eslint/no-empty-function': 'off', 125 | '@typescript-eslint/no-empty-interface': 'error', 126 | '@typescript-eslint/no-explicit-any': 'off', 127 | '@typescript-eslint/no-inferrable-types': [ 128 | 'error', 129 | { 130 | ignoreParameters: true, 131 | }, 132 | ], 133 | '@typescript-eslint/no-misused-new': 'error', 134 | '@typescript-eslint/no-namespace': 'error', 135 | '@typescript-eslint/no-non-null-assertion': 'error', 136 | '@typescript-eslint/no-parameter-properties': 'off', 137 | '@typescript-eslint/no-shadow': [ 138 | 'error', 139 | { 140 | hoist: 'all', 141 | }, 142 | ], 143 | '@typescript-eslint/no-unused-expressions': 'error', 144 | '@typescript-eslint/no-use-before-define': 'off', 145 | '@typescript-eslint/no-var-requires': 'off', 146 | '@typescript-eslint/prefer-for-of': 'error', 147 | '@typescript-eslint/prefer-function-type': 'error', 148 | '@typescript-eslint/prefer-namespace-keyword': 'error', 149 | '@typescript-eslint/quotes': ['error', 'single'], 150 | '@typescript-eslint/semi': ['error', 'always'], 151 | '@typescript-eslint/triple-slash-reference': [ 152 | 'error', 153 | { 154 | path: 'always', 155 | types: 'prefer-import', 156 | lib: 'always', 157 | }, 158 | ], 159 | '@typescript-eslint/type-annotation-spacing': 'error', 160 | '@typescript-eslint/unified-signatures': 'error', 161 | 'arrow-body-style': 'error', 162 | complexity: 'off', 163 | 'constructor-super': 'error', 164 | curly: 'error', 165 | 'dot-notation': 'error', 166 | 'eol-last': 'error', 167 | eqeqeq: ['error', 'smart'], 168 | 'guard-for-in': 'error', 169 | 'id-denylist': ['error', 'any', 'Number', 'number', 'String', 'string', 'Boolean', 'boolean', 'Undefined', 'undefined'], 170 | 'id-match': 'error', 171 | 'import/no-deprecated': 'warn', 172 | indent: 'error', 173 | 'jsdoc/check-alignment': 'error', 174 | 'jsdoc/check-indentation': 'error', 175 | 'jsdoc/newline-after-description': 'error', 176 | 'jsdoc/no-types': 'error', 177 | 'max-classes-per-file': 'off', 178 | 'max-len': [ 179 | 'error', 180 | { 181 | code: 140, 182 | }, 183 | ], 184 | 'new-parens': 'error', 185 | 'no-bitwise': 'error', 186 | 'no-caller': 'error', 187 | 'no-cond-assign': 'error', 188 | 'no-console': [ 189 | 'error', 190 | { 191 | allow: [ 192 | 'log', 193 | 'warn', 194 | 'dir', 195 | 'timeLog', 196 | 'assert', 197 | 'clear', 198 | 'count', 199 | 'countReset', 200 | 'group', 201 | 'groupEnd', 202 | 'table', 203 | 'dirxml', 204 | 'error', 205 | 'groupCollapsed', 206 | 'Console', 207 | 'profile', 208 | 'profileEnd', 209 | 'timeStamp', 210 | 'context', 211 | ], 212 | }, 213 | ], 214 | 'no-debugger': 'error', 215 | 'no-empty': 'off', 216 | 'no-empty-function': 'off', 217 | 'no-eval': 'error', 218 | 'no-fallthrough': 'error', 219 | 'no-invalid-this': 'off', 220 | 'no-new-wrappers': 'error', 221 | 'no-restricted-imports': ['error', 'rxjs/Rx'], 222 | 'no-shadow': 'error', 223 | 'no-throw-literal': 'error', 224 | 'no-trailing-spaces': 'error', 225 | 'no-undef-init': 'error', 226 | 'no-underscore-dangle': 'error', 227 | 'no-unsafe-finally': 'error', 228 | 'no-unused-expressions': 'error', 229 | 'no-unused-labels': 'error', 230 | 'no-use-before-define': 'off', 231 | 'no-var': 'error', 232 | 'object-shorthand': 'error', 233 | 'one-var': ['error', 'never'], 234 | 'prefer-arrow/prefer-arrow-functions': 'error', 235 | 'prefer-const': 'error', 236 | 'quote-props': ['error', 'as-needed'], 237 | quotes: 'error', 238 | radix: 'error', 239 | semi: 'error', 240 | 'space-before-function-paren': [ 241 | 'error', 242 | { 243 | anonymous: 'never', 244 | asyncArrow: 'always', 245 | named: 'never', 246 | }, 247 | ], 248 | 'spaced-comment': [ 249 | 'error', 250 | 'always', 251 | { 252 | markers: ['/'], 253 | }, 254 | ], 255 | 'use-isnan': 'error', 256 | 'valid-typeof': 'off', 257 | '@typescript-eslint/tslint/config': [ 258 | 'error', 259 | { 260 | rules: { 261 | 'import-spacing': true, 262 | typedef: [true, 'call-signature'], 263 | whitespace: [true, 'check-branch', 'check-decl', 'check-operator', 'check-separator', 'check-type', 'check-typecast'], 264 | }, 265 | }, 266 | ], 267 | }, 268 | }; 269 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/.eslintrc.js: -------------------------------------------------------------------------------- 1 | /* 2 | 👋 Hi! This file was autogenerated by tslint-to-eslint-config. 3 | https://github.com/typescript-eslint/tslint-to-eslint-config 4 | 5 | It represents the closest reasonable ESLint configuration to this 6 | project's original TSLint configuration. 7 | 8 | We recommend eventually switching this configuration to extend from 9 | the recommended rulesets in typescript-eslint. 10 | https://github.com/typescript-eslint/tslint-to-eslint-config/blob/master/docs/FAQs.md 11 | 12 | Happy linting! 💖 13 | */ 14 | module.exports = { 15 | env: { 16 | browser: true, 17 | es6: true, 18 | node: true, 19 | }, 20 | extends: ['plugin:@typescript-eslint/recommended', 'plugin:@typescript-eslint/recommended-requiring-type-checking'], 21 | parser: '@typescript-eslint/parser', 22 | parserOptions: { 23 | project: 'tsconfig.json', 24 | sourceType: 'module', 25 | }, 26 | plugins: [ 27 | 'eslint-plugin-import', 28 | 'eslint-plugin-jsdoc', 29 | '@angular-eslint/eslint-plugin', 30 | '@angular-eslint/eslint-plugin-template', 31 | 'eslint-plugin-prefer-arrow', 32 | '@typescript-eslint', 33 | '@typescript-eslint/tslint', 34 | ], 35 | rules: { 36 | '@angular-eslint/component-class-suffix': 'error', 37 | '@angular-eslint/component-selector': [ 38 | 'error', 39 | { 40 | type: 'element', 41 | prefix: 'app', 42 | style: 'kebab-case', 43 | }, 44 | ], 45 | '@angular-eslint/contextual-lifecycle': 'error', 46 | '@angular-eslint/directive-class-suffix': 'error', 47 | '@angular-eslint/directive-selector': [ 48 | 'error', 49 | { 50 | type: 'attribute', 51 | prefix: 'app', 52 | style: 'camelCase', 53 | }, 54 | ], 55 | '@angular-eslint/no-conflicting-lifecycle': 'error', 56 | '@angular-eslint/no-host-metadata-property': 'error', 57 | '@angular-eslint/no-input-rename': 'error', 58 | '@angular-eslint/no-inputs-metadata-property': 'error', 59 | '@angular-eslint/no-output-native': 'error', 60 | '@angular-eslint/no-output-on-prefix': 'error', 61 | '@angular-eslint/no-output-rename': 'error', 62 | '@angular-eslint/no-outputs-metadata-property': 'error', 63 | '@angular-eslint/template/banana-in-box': 'error', 64 | '@angular-eslint/template/eqeqeq': 'error', 65 | '@angular-eslint/template/no-negated-async': 'error', 66 | '@angular-eslint/use-lifecycle-interface': 'error', 67 | '@angular-eslint/use-pipe-transform-interface': 'error', 68 | '@typescript-eslint/adjacent-overload-signatures': 'error', 69 | '@typescript-eslint/array-type': 'off', 70 | '@typescript-eslint/ban-types': [ 71 | 'error', 72 | { 73 | types: { 74 | Object: { 75 | message: 'Avoid using the `Object` type. Did you mean `object`?', 76 | }, 77 | Function: { 78 | message: 'Avoid using the `Function` type. Prefer a specific function type, like `() => void`.', 79 | }, 80 | Boolean: { 81 | message: 'Avoid using the `Boolean` type. Did you mean `boolean`?', 82 | }, 83 | Number: { 84 | message: 'Avoid using the `Number` type. Did you mean `number`?', 85 | }, 86 | String: { 87 | message: 'Avoid using the `String` type. Did you mean `string`?', 88 | }, 89 | Symbol: { 90 | message: 'Avoid using the `Symbol` type. Did you mean `symbol`?', 91 | }, 92 | }, 93 | }, 94 | ], 95 | '@typescript-eslint/consistent-type-assertions': 'error', 96 | '@typescript-eslint/dot-notation': 'error', 97 | '@typescript-eslint/indent': [ 98 | 'error', 99 | 4, 100 | { 101 | FunctionDeclaration: { 102 | parameters: 'first', 103 | }, 104 | FunctionExpression: { 105 | parameters: 'first', 106 | }, 107 | }, 108 | ], 109 | '@typescript-eslint/member-delimiter-style': [ 110 | 'error', 111 | { 112 | multiline: { 113 | delimiter: 'semi', 114 | requireLast: true, 115 | }, 116 | singleline: { 117 | delimiter: 'semi', 118 | requireLast: false, 119 | }, 120 | }, 121 | ], 122 | '@typescript-eslint/member-ordering': 'error', 123 | '@typescript-eslint/naming-convention': 'error', 124 | '@typescript-eslint/no-empty-function': 'off', 125 | '@typescript-eslint/no-empty-interface': 'error', 126 | '@typescript-eslint/no-explicit-any': 'off', 127 | '@typescript-eslint/no-inferrable-types': [ 128 | 'error', 129 | { 130 | ignoreParameters: true, 131 | }, 132 | ], 133 | '@typescript-eslint/no-misused-new': 'error', 134 | '@typescript-eslint/no-namespace': 'error', 135 | '@typescript-eslint/no-non-null-assertion': 'error', 136 | '@typescript-eslint/no-parameter-properties': 'off', 137 | '@typescript-eslint/no-shadow': [ 138 | 'error', 139 | { 140 | hoist: 'all', 141 | }, 142 | ], 143 | '@typescript-eslint/no-unused-expressions': 'error', 144 | '@typescript-eslint/no-use-before-define': 'off', 145 | '@typescript-eslint/no-var-requires': 'off', 146 | '@typescript-eslint/prefer-for-of': 'error', 147 | '@typescript-eslint/prefer-function-type': 'error', 148 | '@typescript-eslint/prefer-namespace-keyword': 'error', 149 | '@typescript-eslint/quotes': ['error', 'single'], 150 | '@typescript-eslint/semi': ['error', 'always'], 151 | '@typescript-eslint/triple-slash-reference': [ 152 | 'error', 153 | { 154 | path: 'always', 155 | types: 'prefer-import', 156 | lib: 'always', 157 | }, 158 | ], 159 | '@typescript-eslint/type-annotation-spacing': 'error', 160 | '@typescript-eslint/unified-signatures': 'error', 161 | 'arrow-body-style': 'error', 162 | complexity: 'off', 163 | 'constructor-super': 'error', 164 | curly: 'error', 165 | 'dot-notation': 'error', 166 | 'eol-last': 'error', 167 | eqeqeq: ['error', 'smart'], 168 | 'guard-for-in': 'error', 169 | 'id-denylist': ['error', 'any', 'Number', 'number', 'String', 'string', 'Boolean', 'boolean', 'Undefined', 'undefined'], 170 | 'id-match': 'error', 171 | 'import/no-deprecated': 'warn', 172 | indent: 'error', 173 | 'jsdoc/check-alignment': 'error', 174 | 'jsdoc/check-indentation': 'error', 175 | 'jsdoc/newline-after-description': 'error', 176 | 'jsdoc/no-types': 'error', 177 | 'max-classes-per-file': 'off', 178 | 'max-len': [ 179 | 'error', 180 | { 181 | code: 140, 182 | }, 183 | ], 184 | 'new-parens': 'error', 185 | 'no-bitwise': 'error', 186 | 'no-caller': 'error', 187 | 'no-cond-assign': 'error', 188 | 'no-console': [ 189 | 'error', 190 | { 191 | allow: [ 192 | 'log', 193 | 'warn', 194 | 'dir', 195 | 'timeLog', 196 | 'assert', 197 | 'clear', 198 | 'count', 199 | 'countReset', 200 | 'group', 201 | 'groupEnd', 202 | 'table', 203 | 'dirxml', 204 | 'error', 205 | 'groupCollapsed', 206 | 'Console', 207 | 'profile', 208 | 'profileEnd', 209 | 'timeStamp', 210 | 'context', 211 | ], 212 | }, 213 | ], 214 | 'no-debugger': 'error', 215 | 'no-empty': 'off', 216 | 'no-empty-function': 'off', 217 | 'no-eval': 'error', 218 | 'no-fallthrough': 'error', 219 | 'no-invalid-this': 'off', 220 | 'no-new-wrappers': 'error', 221 | 'no-restricted-imports': ['error', 'rxjs/Rx'], 222 | 'no-shadow': 'error', 223 | 'no-throw-literal': 'error', 224 | 'no-trailing-spaces': 'error', 225 | 'no-undef-init': 'error', 226 | 'no-underscore-dangle': 'error', 227 | 'no-unsafe-finally': 'error', 228 | 'no-unused-expressions': 'error', 229 | 'no-unused-labels': 'error', 230 | 'no-use-before-define': 'off', 231 | 'no-var': 'error', 232 | 'object-shorthand': 'error', 233 | 'one-var': ['error', 'never'], 234 | 'prefer-arrow/prefer-arrow-functions': 'error', 235 | 'prefer-const': 'error', 236 | 'quote-props': ['error', 'as-needed'], 237 | quotes: 'error', 238 | radix: 'error', 239 | semi: 'error', 240 | 'space-before-function-paren': [ 241 | 'error', 242 | { 243 | anonymous: 'never', 244 | asyncArrow: 'always', 245 | named: 'never', 246 | }, 247 | ], 248 | 'spaced-comment': [ 249 | 'error', 250 | 'always', 251 | { 252 | markers: ['/'], 253 | }, 254 | ], 255 | 'use-isnan': 'error', 256 | 'valid-typeof': 'off', 257 | '@typescript-eslint/tslint/config': [ 258 | 'error', 259 | { 260 | rules: { 261 | 'import-spacing': true, 262 | typedef: [true, 'call-signature'], 263 | whitespace: [true, 'check-branch', 'check-decl', 'check-operator', 'check-separator', 'check-type', 'check-typecast'], 264 | }, 265 | }, 266 | ], 267 | }, 268 | }; 269 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/lib/pinch-zoom.component.ts: -------------------------------------------------------------------------------- 1 | import { 2 | Component, 3 | ElementRef, 4 | EventEmitter, 5 | HostBinding, 6 | Input, 7 | OnChanges, 8 | OnDestroy, 9 | OnInit, Output, 10 | SimpleChanges 11 | } from '@angular/core'; 12 | 13 | import { Properties } from './interfaces'; 14 | import { defaultProperties, backwardCompatibilityProperties } from './properties'; 15 | import { IvyPinch } from './ivypinch'; 16 | 17 | 18 | interface ComponentProperties extends Properties { 19 | disabled?: boolean; 20 | overflow?: 'hidden' | 'visible'; 21 | disableZoomControl?: 'disable' | 'never' | 'auto'; 22 | backgroundColor?: string; 23 | } 24 | 25 | export const _defaultComponentProperties: ComponentProperties = { 26 | overflow: 'hidden', 27 | disableZoomControl: 'auto', 28 | backgroundColor: 'rgba(0,0,0,0.85)', 29 | }; 30 | 31 | @Component({ 32 | selector: 'pinch-zoom, [pinch-zoom]', 33 | exportAs: 'pinchZoom', 34 | templateUrl: './pinch-zoom.component.html', 35 | styleUrls: ['./pinch-zoom.component.sass'], 36 | standalone: true, 37 | imports: [] 38 | }) 39 | export class PinchZoomComponent implements OnInit, OnDestroy, OnChanges { 40 | private pinchZoom: IvyPinch; 41 | private _properties!: ComponentProperties; 42 | private readonly defaultComponentProperties!: ComponentProperties; 43 | private _transitionDuration!: number; 44 | private _doubleTap!: boolean; 45 | private _doubleTapScale!: number; 46 | private _autoZoomOut!: boolean; 47 | private _limitZoom!: number | 'original image size'; 48 | 49 | @Input('properties') set properties(value: ComponentProperties) { 50 | if (value) { 51 | this._properties = value; 52 | } 53 | } 54 | 55 | get properties(): ComponentProperties { 56 | return this._properties; 57 | } 58 | 59 | // transitionDuration 60 | @Input('transition-duration') set transitionDurationBackwardCompatibility(value: number) { 61 | if (value) { 62 | this._transitionDuration = value; 63 | } 64 | } 65 | 66 | @Input('transitionDuration') set transitionDuration(value: number) { 67 | if (value) { 68 | this._transitionDuration = value; 69 | } 70 | } 71 | 72 | get transitionDuration(): number { 73 | return this._transitionDuration; 74 | } 75 | 76 | // doubleTap 77 | @Input('double-tap') set doubleTapBackwardCompatibility(value: boolean) { 78 | if (value) { 79 | this._doubleTap = value; 80 | } 81 | } 82 | 83 | @Input('doubleTap') set doubleTap(value: boolean) { 84 | if (value) { 85 | this._doubleTap = value; 86 | } 87 | } 88 | 89 | get doubleTap(): boolean { 90 | return this._doubleTap; 91 | } 92 | 93 | // doubleTapScale 94 | @Input('double-tap-scale') set doubleTapScaleBackwardCompatibility(value: number) { 95 | if (value) { 96 | this._doubleTapScale = value; 97 | } 98 | } 99 | 100 | @Input('doubleTapScale') set doubleTapScale(value: number) { 101 | if (value) { 102 | this._doubleTapScale = value; 103 | } 104 | } 105 | 106 | get doubleTapScale(): number { 107 | return this._doubleTapScale; 108 | } 109 | 110 | // autoZoomOut 111 | @Input('auto-zoom-out') set autoZoomOutBackwardCompatibility(value: boolean) { 112 | if (value) { 113 | this._autoZoomOut = value; 114 | } 115 | } 116 | 117 | @Input('autoZoomOut') set autoZoomOut(value: boolean) { 118 | if (value) { 119 | this._autoZoomOut = value; 120 | } 121 | } 122 | 123 | get autoZoomOut(): boolean { 124 | return this._autoZoomOut; 125 | } 126 | 127 | // limitZoom 128 | @Input('limit-zoom') set limitZoomBackwardCompatibility(value: number | 'original image size') { 129 | if (value) { 130 | this._limitZoom = value; 131 | } 132 | } 133 | 134 | @Input('limitZoom') set limitZoom(value: number | 'original image size') { 135 | if (value) { 136 | this._limitZoom = value; 137 | } 138 | } 139 | 140 | get limitZoom(): number | 'original image size' { 141 | return this._limitZoom; 142 | } 143 | 144 | @Input() disabled!: boolean; 145 | @Input() disablePan!: boolean; 146 | @Input() overflow!: 'hidden' | 'visible'; 147 | @Input() zoomControlScale!: number; 148 | @Input() disableZoomControl!: 'disable' | 'never' | 'auto'; 149 | @Input() backgroundColor!: string; 150 | @Input() limitPan!: boolean; 151 | @Input() minPanScale!: number; 152 | @Input() minScale!: number; 153 | @Input() listeners!: 'auto' | 'mouse and touch'; 154 | @Input() wheel!: boolean; 155 | @Input() autoHeight!: boolean; 156 | @Input() wheelZoomFactor!: number; 157 | @Input() draggableImage!: boolean; 158 | @Input() draggableOnPinch!: boolean; 159 | @Output() public zoomChanged: EventEmitter = new EventEmitter(); 160 | 161 | @HostBinding('style.overflow') 162 | get hostOverflow(): 'hidden' | 'visible' { 163 | return this.properties['overflow']; 164 | } 165 | 166 | @HostBinding('style.background-color') 167 | get hostBackgroundColor(): string { 168 | return this.properties['backgroundColor']; 169 | } 170 | 171 | get isTouchScreen(): boolean { 172 | const prefixes = ' -webkit- -moz- -o- -ms- '.split(' '); 173 | const mq = (query: string): boolean => { 174 | return window.matchMedia(query).matches; 175 | }; 176 | 177 | if ('ontouchstart' in window) { 178 | return true; 179 | } 180 | 181 | // include the 'heartz' as a way to have a non matching MQ to help terminate the join 182 | // https://git.io/vznFH 183 | const query = ['(', prefixes.join('touch-enabled),('), 'heartz', ')'].join(''); 184 | return mq(query); 185 | } 186 | 187 | get isDragging(): boolean { 188 | return this.pinchZoom?.isDragging(); 189 | } 190 | 191 | get isDisabled(): boolean { 192 | return this._properties.disabled; 193 | } 194 | 195 | get scale(): number { 196 | return this.pinchZoom.scale; 197 | } 198 | 199 | get isZoomedIn(): boolean { 200 | return this.scale > 1; 201 | } 202 | 203 | get scaleLevel(): number { 204 | return Math.round(this.scale / this._zoomControlScale); 205 | } 206 | 207 | get maxScale(): number { 208 | return this.pinchZoom.maxScale; 209 | } 210 | 211 | get isZoomLimitReached(): boolean { 212 | return this.scale >= this.maxScale; 213 | } 214 | 215 | get _zoomControlScale(): number { 216 | return this.getPropertiesValue('zoomControlScale'); 217 | } 218 | 219 | constructor(private elementRef: ElementRef) { 220 | this.defaultComponentProperties = this.getDefaultComponentProperties(); 221 | this.applyPropertiesDefault(this.defaultComponentProperties, {}); 222 | } 223 | 224 | ngOnInit(): void { 225 | this.initPinchZoom(); 226 | 227 | /* Calls the method until the image size is available */ 228 | this.detectLimitZoom(); 229 | } 230 | 231 | ngOnChanges(changes: SimpleChanges): void { 232 | let changedProperties = this.getProperties(changes); 233 | changedProperties = this.renameProperties(changedProperties); 234 | 235 | this.applyPropertiesDefault(this.defaultComponentProperties, changedProperties); 236 | } 237 | 238 | ngOnDestroy(): void { 239 | this.destroy(); 240 | } 241 | 242 | private initPinchZoom(): void { 243 | if (this._properties.disabled) { 244 | return; 245 | } 246 | 247 | this._properties.limitZoom = this.limitZoom; 248 | this._properties.element = this.elementRef.nativeElement.querySelector('.pinch-zoom-content'); 249 | this.pinchZoom = new IvyPinch(this.properties, this.zoomChanged); 250 | } 251 | 252 | private getProperties( 253 | changes: SimpleChanges, 254 | ): ComponentProperties | Record { 255 | let properties: ComponentProperties = {}; 256 | 257 | for (const prop in changes) { 258 | if (prop !== 'properties') { 259 | properties[prop] = changes[prop].currentValue; 260 | } 261 | if (prop === 'properties') { 262 | properties = changes[prop].currentValue; 263 | } 264 | } 265 | return properties; 266 | } 267 | 268 | private renameProperties( 269 | properties: ComponentProperties | Record, 270 | ): ComponentProperties { 271 | for (const prop in properties) { 272 | if (backwardCompatibilityProperties[prop]) { 273 | properties[backwardCompatibilityProperties[prop]] = properties[prop]; 274 | delete properties[prop]; 275 | } 276 | } 277 | 278 | return properties as ComponentProperties; 279 | } 280 | 281 | private applyPropertiesDefault(defaultProperties: ComponentProperties, properties: ComponentProperties): void { 282 | this.properties = Object.assign({}, defaultProperties, properties); 283 | } 284 | 285 | toggleZoom(): void { 286 | this.pinchZoom?.toggleZoom(); 287 | } 288 | 289 | zoomIn(value: number): number { 290 | return this.pinchZoom?.zoomIn(value); 291 | } 292 | 293 | zoomOut(value: number): number { 294 | return this.pinchZoom?.zoomOut(value); 295 | } 296 | 297 | isControl(): boolean { 298 | if (this.isDisabled) { 299 | return false; 300 | } 301 | 302 | if (this._properties.disableZoomControl === 'disable') { 303 | return false; 304 | } 305 | 306 | if (this.isTouchScreen && this._properties.disableZoomControl === 'auto') { 307 | return false; 308 | } 309 | 310 | return true; 311 | } 312 | 313 | detectLimitZoom(): void { 314 | this.pinchZoom?.detectLimitZoom(); 315 | } 316 | 317 | destroy(): void { 318 | this.pinchZoom?.destroy(); 319 | } 320 | 321 | private getPropertiesValue(propertyName: K): ComponentProperties[K] { 322 | if (this.properties && this.properties[propertyName]) { 323 | return this.properties[propertyName]; 324 | } else { 325 | return this.defaultComponentProperties[propertyName]; 326 | } 327 | } 328 | 329 | private getDefaultComponentProperties(): ComponentProperties { 330 | return { ...defaultProperties, ..._defaultComponentProperties }; 331 | } 332 | } 333 | -------------------------------------------------------------------------------- /src/app/app.component.html: -------------------------------------------------------------------------------- 1 |
2 |

Properties

3 | 4 | 5 |
6 |

transition-duration: 1000

7 |

Defines the speed of the animation of positioning and transforming.

8 |

Zoom State: {{this.zoomstate}}

9 | 10 | 13 | 14 |
15 | 16 | 17 |
18 |

limit-zoom: 2

19 |

Limit the maximum available scale. By default, the maximum scale is calculated based on the original image size.

20 |

Zoom State: {{this.zoomstate}}

21 | 22 | 25 | 26 |
27 | 28 | 29 |
30 |

minScale: 1

31 |

Limit the minimum acceptable scale. With a value of 1, it is recommended to use this parameter with limitPan.

32 |

Zoom State: {{this.zoomstate}}

33 | 34 | 37 | 38 |
39 | 40 | 41 |
42 |

auto-zoom-out: true

43 |

Automatic restoration of the original size of an image after its zooming in by two fingers.

44 |

Zoom State: {{this.zoomstate}}

45 | 46 | 49 | 50 |
51 | 52 | 53 |
54 |

double-tap: false

55 |

Zooming in and zooming out of an image, depending on its current condition, with double tap.

56 |

Zoom State: {{this.zoomstate}}

57 | 58 | 61 | 62 |
63 | 64 | 65 |
66 |

double-tap-scale: 4

67 |

Double tap scaling factor.

68 |

Zoom State: {{this.zoomstate}}

69 | 70 | 73 | 74 |
75 | 76 | 77 |
78 |

disabled: true

79 |

Zoom State: {{this.zoomstate}}

80 | 81 | 84 | 85 |
86 | 87 | 88 |
89 |

disablePan: true

90 |

Turn off panning with one finger.

91 |

Zoom State: {{this.zoomstate}}

92 | 93 | 96 | 97 |
98 | 99 | 100 |
101 |

minPanScale: 2

102 |

Minimum zoom at which panning is enabled.

103 |

Zoom State: {{this.zoomstate}}

104 | 105 | 108 | 109 |
110 | 111 | 112 |
113 |

overflow: 'visible'

114 |

115 | hidden - the overflow is clipped, and the rest of the content will be invisible. visible - the overflow is not clipped. The 116 | content renders outside the element's box. 117 |

118 |

Zoom State: {{this.zoomstate}}

119 | 120 | 123 | 124 |
125 | 126 | 127 |
128 |

disableZoomControl: 'disable'

129 |

130 | Disable zoom controls. auto - Disable zoom controls on touch screen devices. never - show zoom controls on all devices. disable 131 | - disable zoom controls on all devices. 132 |

133 |

Zoom State: {{this.zoomstate}}

134 | 135 | 138 | 139 |
140 | 141 | 142 |
143 |

zoomControlScale: 2

144 |

Zoom factor when using zoom controls.

145 |

Zoom State: {{this.zoomstate}}

146 | 147 | 150 | 151 |
152 | 153 | 154 |
155 |

backgroundColor: 'rgba(0,0,0,0.65)'

156 |

The background color of the container.

157 |

Zoom State: {{this.zoomstate}}

158 | 159 | 162 | 163 |
164 | 165 | 166 |
167 |

limitPan: true

168 |

Stop panning when the edge of the image reaches the edge of the screen.

169 |

Zoom State: {{this.zoomstate}}

170 | 171 | 174 | 175 |
176 | 177 | 178 |
179 |

listeners: 'auto'

180 |

181 | By default, subscriptions are made for mouse and touch screen events. The value auto means that the subscription will be only 182 | for touch events or only for mouse events, depending on the type of screen. 183 |

184 |

Zoom State: {{this.zoomstate}}

185 | 186 | 189 | 190 |
191 | 192 | 193 |
194 |

wheel: false

195 |

Scale with the mouse wheel.

196 |

Zoom State: {{this.zoomstate}}

197 | 198 | 201 | 202 |
203 | 204 | 205 |
206 |

wheelZoomFactor: 0.5

207 |

Zoom factor when zoomed in with the mouse wheel.

208 |

Zoom State: {{this.zoomstate}}

209 | 210 | 213 | 214 |
215 | 216 | 217 |
218 |

autoHeight: true

219 |

220 | Calculate the height of the container based on the width and height attributes of the image. By default, the width of the 221 | container is 100%, and the height is determined after the image information is loaded - this may cause a delay in determining 222 | the height of the container. If you want the container to initially have dimensions corresponding to the dimensions of the 223 | image, then specify the attributes width and height for the tag. When setting the property value to `true`, a 224 | subscription to the window resize listener will be created. 225 |

226 |

Zoom State: {{this.zoomstate}}

227 | 228 | 233 | 234 |
235 | 236 | 237 |
238 |

draggableImage: true

239 |

Sets the attribute draggable to the img tag.

240 |

Zoom State: {{this.zoomstate}}

241 | 242 | 245 | 246 |
247 | 248 | 249 |
250 |

draggableOnPinch: true

251 |

Zoom State: {{this.zoomstate}}

252 | 253 | 256 | 257 |
258 | 259 | 260 |

Methods

261 |
262 | 263 | 266 | 267 |
268 | 269 | 270 |
271 |
272 |
273 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Note 2 | 3 | This Project has been forked to be updated to work with Angular versions 19 and 20. 4 | 5 | # Pinch zoom for Angular 6 | 7 | The module provides opportunities for image zooming in, zooming out and positioning with use of gestures on a touch screen. 8 | 9 | ## Installation 10 | 11 | Install the npm package. 12 | 13 | ``` 14 | npm i @meddv/ngx-pinch-zoom 15 | ``` 16 | 17 | Import module: 18 | 19 | ```ts 20 | import { PinchZoomComponent } from '@meddv/ngx-pinch-zoom'; 21 | 22 | @NgModule({ 23 | imports: [ PinchZoomComponent ] 24 | }) 25 | ``` 26 | 27 | ## Usage 28 | 29 | For use, put your image inside the <pinch-zoom> container. Please, pay attention to the parameters of your viewport metatag. If you use Pinch Zoom, it is required to limit zooming of a web-page, by entering the following parameters: <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1.0, user-scalable=no">. 30 | 31 | ```html 32 | 33 | 34 | 35 | ``` 36 | 37 | ## Properties 38 | 39 | | name | type | default | description | 40 | | ------------------- | ----------------------------- | --------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | 41 | | transition-duration | number | 200 | Defines the speed of the animation of positioning and transforming. | 42 | | limit-zoom | number, "original image size" | "original image size" | Limit the maximum available scale. By default, the maximum scale is calculated based on the original image size. | 43 | | minScale | number | 0 | Limit the minimum acceptable scale. With a value of 1, it is recommended to use this parameter with `limitPan` | 44 | | auto-zoom-out | boolean | false | Automatic restoration of the original size of an image after its zooming in by two fingers. | 45 | | double-tap | boolean | true | Zooming in and zooming out of an image, depending on its current condition, with double tap. | 46 | | disabled | boolean | false | Disable zoom. | 47 | | disablePan | boolean | false | Turn off panning with one finger. | 48 | | overflow | "hidden", "visible" | "hidden" | `hidden` - the overflow is clipped, and the rest of the content will be invisible. `visible` - the overflow is not clipped. The content renders outside the element's box. | 49 | | disableZoomControl | "disable", "never", "auto" | "auto" | Disable zoom controls. `auto` - Disable zoom controls on touch screen devices. `never` - show zoom controls on all devices. `disable` - disable zoom controls on all devices. | 50 | | zoomControlScale | number | 1 | Zoom factor when using zoom controls. | 51 | | backgroundColor | string | "rgba(0,0,0,0.85)" | The background color of the container. | 52 | | limitPan | boolean | false | Stop panning when the edge of the image reaches the edge of the screen. | 53 | | minPanScale | number | 1.0001 | Minimum zoom at which panning is enabled. | 54 | | listeners | "auto", "mouse and touch" | "mouse and touch" | By default, subscriptions are made for mouse and touch screen events. The value `auto` means that the subscription will be only for touch events or only for mouse events, depending on the type of screen. | 55 | | wheel | boolean | true | Scale with the mouse wheel. | 56 | | wheelZoomFactor | number | 0.2 | Zoom factor when zoomed in with the mouse wheel. | 57 | | autoHeight | boolean | false | Calculate the height of the container based on the `width` and `height` attributes of the image. By default, the width of the container is 100%, and the height is determined after the image information is loaded - this may cause a delay in determining the height of the container. If you want the container to initially have dimensions corresponding to the dimensions of the image, then specify the attributes `width` and `height` for the `` tag. When setting the property value to `true`, a subscription to the window resize listener will be created. | 58 | | draggableImage | boolean | false | Sets the attribute `draggable` to the `` tag. | 59 | | draggableOnPinch | boolean | false | When set to `true` content can be moved around while touching or pinching with two fingers. | 60 | 61 | ## Outputs 62 | 63 | | name | description | 64 | | ----------- | ---------------------------------------------- | 65 | | zoomChanged | Emits current `scale: number` if it's changed. | 66 | 67 | ## Methods 68 | 69 | | name | description | 70 | | ---------------------- | -------------------------------------------------------------------------------------------- | 71 | | toggleZoom() | Image zooming in and out, depending on its current state. | 72 | | zoomIn(value: number) | Zoom in by `value`, respects `limit-zoom` option. Returns `scale: number`. | 73 | | zoomOut(value: number) | Zoom out by `value`, respects `minScale` option. Returns `scale: number`. | 74 | | destroy() | Unsubscribe from mouse events and touches, as well as remove added styles from the DOM tree. | 75 | 76 | ## Contributor services 77 | 78 | Contact us over our Issue Tracker. 79 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/README.md: -------------------------------------------------------------------------------- 1 | # Note 2 | 3 | This Project has been forked to be updated to work with Angular versions 19 and 20. 4 | 5 | # Pinch zoom for Angular 6 | 7 | The module provides opportunities for image zooming in, zooming out and positioning with use of gestures on a touch screen. 8 | 9 | ## Installation 10 | 11 | Install the npm package. 12 | 13 | ``` 14 | npm i @meddv/ngx-pinch-zoom 15 | ``` 16 | 17 | Import module: 18 | 19 | ```ts 20 | import { PinchZoomComponent } from '@meddv/ngx-pinch-zoom'; 21 | 22 | @NgModule({ 23 | imports: [ PinchZoomComponent ] 24 | }) 25 | ``` 26 | 27 | ## Usage 28 | 29 | For use, put your image inside the <pinch-zoom> container. Please, pay attention to the parameters of your viewport metatag. If you use Pinch Zoom, it is required to limit zooming of a web-page, by entering the following parameters: <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1.0, user-scalable=no">. 30 | 31 | ```html 32 | 33 | 34 | 35 | ``` 36 | 37 | ## Properties 38 | 39 | | name | type | default | description | 40 | | ------------------- | ----------------------------- | --------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | 41 | | transition-duration | number | 200 | Defines the speed of the animation of positioning and transforming. | 42 | | limit-zoom | number, "original image size" | "original image size" | Limit the maximum available scale. By default, the maximum scale is calculated based on the original image size. | 43 | | minScale | number | 0 | Limit the minimum acceptable scale. With a value of 1, it is recommended to use this parameter with `limitPan` | 44 | | auto-zoom-out | boolean | false | Automatic restoration of the original size of an image after its zooming in by two fingers. | 45 | | double-tap | boolean | true | Zooming in and zooming out of an image, depending on its current condition, with double tap. | 46 | | disabled | boolean | false | Disable zoom. | 47 | | disablePan | boolean | false | Turn off panning with one finger. | 48 | | overflow | "hidden", "visible" | "hidden" | `hidden` - the overflow is clipped, and the rest of the content will be invisible. `visible` - the overflow is not clipped. The content renders outside the element's box. | 49 | | disableZoomControl | "disable", "never", "auto" | "auto" | Disable zoom controls. `auto` - Disable zoom controls on touch screen devices. `never` - show zoom controls on all devices. `disable` - disable zoom controls on all devices. | 50 | | zoomControlScale | number | 1 | Zoom factor when using zoom controls. | 51 | | backgroundColor | string | "rgba(0,0,0,0.85)" | The background color of the container. | 52 | | limitPan | boolean | false | Stop panning when the edge of the image reaches the edge of the screen. | 53 | | minPanScale | number | 1.0001 | Minimum zoom at which panning is enabled. | 54 | | listeners | "auto", "mouse and touch" | "mouse and touch" | By default, subscriptions are made for mouse and touch screen events. The value `auto` means that the subscription will be only for touch events or only for mouse events, depending on the type of screen. | 55 | | wheel | boolean | true | Scale with the mouse wheel. | 56 | | wheelZoomFactor | number | 0.2 | Zoom factor when zoomed in with the mouse wheel. | 57 | | autoHeight | boolean | false | Calculate the height of the container based on the `width` and `height` attributes of the image. By default, the width of the container is 100%, and the height is determined after the image information is loaded - this may cause a delay in determining the height of the container. If you want the container to initially have dimensions corresponding to the dimensions of the image, then specify the attributes `width` and `height` for the `` tag. When setting the property value to `true`, a subscription to the window resize listener will be created. | 58 | | draggableImage | boolean | false | Sets the attribute `draggable` to the `` tag. | 59 | | draggableOnPinch | boolean | false | When set to `true` content can be moved around while touching or pinching with two fingers. | 60 | 61 | ## Outputs 62 | 63 | | name | description | 64 | | ----------- | ---------------------------------------------- | 65 | | zoomChanged | Emits current `scale: number` if it's changed. | 66 | 67 | ## Methods 68 | 69 | | name | description | 70 | | ---------------------- | -------------------------------------------------------------------------------------------- | 71 | | toggleZoom() | Image zooming in and out, depending on its current state. | 72 | | zoomIn(value: number) | Zoom in by `value`, respects `limit-zoom` option. Returns `scale: number`. | 73 | | zoomOut(value: number) | Zoom out by `value`, respects `minScale` option. Returns `scale: number`. | 74 | | destroy() | Unsubscribe from mouse events and touches, as well as remove added styles from the DOM tree. | 75 | 76 | ## Contributor services 77 | 78 | Contact us over our Issue Tracker. 79 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/lib/touches.ts: -------------------------------------------------------------------------------- 1 | export interface Properties { 2 | element: HTMLElement; 3 | listeners?: 'auto' | 'mouse and touch'; 4 | touchListeners?: TouchListeners; 5 | mouseListeners?: MouseListeners; 6 | otherListeners?: OtherListeners; 7 | resize?: boolean; 8 | } 9 | 10 | export type EventType = 11 | | undefined 12 | | 'touchstart' 13 | | 'touchend' 14 | | 'touchmove' 15 | | 'mousedown' 16 | | 'mouseup' 17 | | 'mousemove' 18 | | 'pan' 19 | | 'pinch' 20 | | 'horizontal-swipe' 21 | | 'vertical-swipe' 22 | | 'tap' 23 | | 'longtap' 24 | | 'wheel' 25 | | 'double-tap' 26 | | 'resize'; 27 | export type TouchHandler ='handleTouchstart' | 'handleTouchmove' | 'handleTouchend'; 28 | export type MouseHandler = 'handleMousedown' | 'handleMousemove' | 'handleMouseup' | 'handleWheel'; 29 | export type OtherHandler = 'handleResize'; 30 | 31 | export type TouchListeners = Partial>; 32 | export type MouseListeners = Partial>; 33 | export type OtherListeners = Partial>; 34 | 35 | export class Touches { 36 | private properties: Properties; 37 | private element: HTMLElement; 38 | private elementPosition: DOMRect; 39 | private eventType: EventType = undefined; 40 | private handlers: TouchListeners | MouseListeners | OtherListeners = {}; 41 | private startX = 0; 42 | private startY = 0; 43 | private lastTap = 0; 44 | private doubleTapTimeout: number; 45 | private doubleTapMinTimeout = 300; 46 | private tapMinTimeout = 200; 47 | private touchstartTime = 0; 48 | private i: number = 0; 49 | private isMousedown = false; 50 | 51 | private _touchListeners: Record<'touchstart' | 'touchmove' | 'touchend', TouchHandler> = { 52 | touchstart: 'handleTouchstart', 53 | touchmove: 'handleTouchmove', 54 | touchend: 'handleTouchend', 55 | }; 56 | private _mouseListeners: Record<'mousedown' | 'mousemove' | 'mouseup' | 'wheel', MouseHandler> = { 57 | mousedown: 'handleMousedown', 58 | mousemove: 'handleMousemove', 59 | mouseup: 'handleMouseup', 60 | wheel: 'handleWheel', 61 | }; 62 | private _otherListeners: Record<'resize', OtherHandler> = { 63 | resize: 'handleResize', 64 | }; 65 | 66 | private get touchListeners(): TouchListeners { 67 | return this.properties.touchListeners ? this.properties.touchListeners : this._touchListeners; 68 | } 69 | 70 | private get mouseListeners(): MouseListeners { 71 | return this.properties.mouseListeners ? this.properties.mouseListeners : this._mouseListeners; 72 | } 73 | 74 | private get otherListeners(): OtherListeners { 75 | return this.properties.otherListeners ? this.properties.otherListeners : this._otherListeners; 76 | } 77 | 78 | constructor(properties: Properties) { 79 | this.properties = properties; 80 | this.element = this.properties.element; 81 | this.elementPosition = this.getElementPosition(); 82 | 83 | this.toggleEventListeners('addEventListener'); 84 | } 85 | 86 | public destroy(): void { 87 | this.toggleEventListeners('removeEventListener'); 88 | } 89 | 90 | private toggleEventListeners(action: 'addEventListener' | 'removeEventListener'): void { 91 | let listeners: TouchListeners | MouseListeners | OtherListeners; 92 | 93 | if (this.properties.listeners === 'mouse and touch') { 94 | listeners = Object.assign(this.touchListeners, this.mouseListeners); 95 | } else { 96 | listeners = this.detectTouchScreen() ? this.touchListeners : this.mouseListeners; 97 | } 98 | 99 | if (this.properties.resize) { 100 | listeners = Object.assign(listeners, this.otherListeners); 101 | } 102 | 103 | for (const listener in listeners) { 104 | const handler = listeners[listener]; 105 | 106 | // Window 107 | if (listener === 'resize') { 108 | if (action === 'addEventListener') { 109 | window.addEventListener(listener, this[handler], false); 110 | } 111 | if (action === 'removeEventListener') { 112 | window.removeEventListener(listener, this[handler], false); 113 | } 114 | // Document 115 | } else if (listener === 'mouseup' || listener === 'mousemove') { 116 | if (action === 'addEventListener') { 117 | document.addEventListener(listener, this[handler], false); 118 | } 119 | if (action === 'removeEventListener') { 120 | document.removeEventListener(listener, this[handler], false); 121 | } 122 | // Element 123 | } else { 124 | if (action === 'addEventListener') { 125 | this.element.addEventListener(listener, this[handler], false); 126 | } 127 | if (action === 'removeEventListener') { 128 | this.element.removeEventListener(listener, this[handler], false); 129 | } 130 | } 131 | } 132 | } 133 | 134 | public addEventListeners(listener: string): void { 135 | const handler: MouseHandler = this._mouseListeners[listener]; 136 | window.addEventListener(listener, this[handler], false); 137 | } 138 | 139 | public removeEventListeners(listener: string): void { 140 | const handler: MouseHandler = this._mouseListeners[listener]; 141 | window.removeEventListener(listener, this[handler], false); 142 | } 143 | 144 | /* 145 | * Listeners 146 | */ 147 | 148 | /* Touchstart */ 149 | 150 | private handleTouchstart = (event: TouchEvent): void => { 151 | this.elementPosition = this.getElementPosition(); 152 | this.touchstartTime = new Date().getTime(); 153 | 154 | if (this.eventType === undefined) { 155 | this.getTouchstartPosition(event); 156 | } 157 | 158 | this.runHandler('touchstart', event); 159 | }; 160 | 161 | /* Touchmove */ 162 | 163 | private handleTouchmove = (event: TouchEvent): void => { 164 | const touches = event.touches; 165 | 166 | // Pan 167 | if (this.detectPan(touches)) { 168 | this.runHandler('pan', event); 169 | } 170 | 171 | // Pinch 172 | if (this.detectPinch(event)) { 173 | this.runHandler('pinch', event); 174 | } 175 | }; 176 | 177 | private handleLinearSwipe(event: any): void { 178 | //event.preventDefault(); 179 | 180 | this.i++; 181 | 182 | if (this.i > 3) { 183 | this.eventType = this.getLinearSwipeType(event); 184 | } 185 | 186 | if (this.eventType === 'horizontal-swipe') { 187 | this.runHandler('horizontal-swipe', event); 188 | } 189 | 190 | if (this.eventType === 'vertical-swipe') { 191 | this.runHandler('vertical-swipe', event); 192 | } 193 | } 194 | 195 | /* Touchend */ 196 | 197 | private handleTouchend = (event: TouchEvent): void => { 198 | const touches = event.touches; 199 | 200 | // Double Tap 201 | if (this.detectDoubleTap()) { 202 | this.runHandler('double-tap', event); 203 | } 204 | 205 | // Tap 206 | this.detectTap(); 207 | 208 | this.runHandler('touchend', event); 209 | this.eventType = 'touchend'; 210 | 211 | if (touches && touches.length === 0) { 212 | this.eventType = undefined; 213 | this.i = 0; 214 | } 215 | }; 216 | 217 | /* Mousedown */ 218 | 219 | private handleMousedown = (event: MouseEvent): void => { 220 | this.isMousedown = true; 221 | this.elementPosition = this.getElementPosition(); 222 | this.touchstartTime = new Date().getTime(); 223 | 224 | if (this.eventType === undefined) { 225 | this.getMousedownPosition(event); 226 | } 227 | 228 | this.runHandler('mousedown', event); 229 | }; 230 | 231 | /* Mousemove */ 232 | 233 | private handleMousemove = (event: MouseEvent): void => { 234 | //event.preventDefault(); 235 | 236 | if (!this.isMousedown) { 237 | return; 238 | } 239 | 240 | // Pan 241 | this.runHandler('pan', event); 242 | 243 | // Linear swipe 244 | switch (this.detectLinearSwipe(event)) { 245 | case 'horizontal-swipe': 246 | // FIXME: looks like an error 247 | // @ts-ignore 248 | event.swipeType = 'horizontal-swipe'; 249 | this.runHandler('horizontal-swipe', event); 250 | break; 251 | case 'vertical-swipe': 252 | // FIXME: looks like an error 253 | // @ts-ignore 254 | event.swipeType = 'vertical-swipe'; 255 | this.runHandler('vertical-swipe', event); 256 | break; 257 | } 258 | 259 | // Linear swipe 260 | if ( 261 | this.detectLinearSwipe(event) || 262 | this.eventType === 'horizontal-swipe' || 263 | this.eventType === 'vertical-swipe' 264 | ) { 265 | this.handleLinearSwipe(event); 266 | } 267 | }; 268 | 269 | /* Mouseup */ 270 | 271 | private handleMouseup = (event: MouseEvent): void => { 272 | // Tap 273 | this.detectTap(); 274 | 275 | this.isMousedown = false; 276 | this.runHandler('mouseup', event); 277 | this.eventType = undefined; 278 | this.i = 0; 279 | }; 280 | 281 | /* Wheel */ 282 | 283 | private handleWheel = (event: WheelEvent): void => { 284 | this.runHandler('wheel', event); 285 | }; 286 | 287 | /* Resize */ 288 | 289 | private handleResize = (event: Event): void => { 290 | this.runHandler('resize', event); 291 | }; 292 | 293 | private runHandler(eventName: EventType, event: unknown):void { 294 | if (this.handlers[eventName]) { 295 | this.handlers[eventName](event); 296 | } 297 | } 298 | 299 | /* 300 | * Detection 301 | */ 302 | 303 | private detectPan(touches: TouchList): boolean { 304 | return (touches.length === 1 && !this.eventType) || this.eventType === 'pan'; 305 | } 306 | 307 | private detectDoubleTap(): boolean { 308 | if (this.eventType != undefined) { 309 | return; 310 | } 311 | 312 | const currentTime = new Date().getTime(); 313 | const tapLength = currentTime - this.lastTap; 314 | 315 | window.clearTimeout(this.doubleTapTimeout); 316 | 317 | if (tapLength < this.doubleTapMinTimeout && tapLength > 0) { 318 | return true; 319 | } else { 320 | this.doubleTapTimeout = window.setTimeout(() => { 321 | window.clearTimeout(this.doubleTapTimeout); 322 | }, this.doubleTapMinTimeout); 323 | } 324 | this.lastTap = currentTime; 325 | 326 | return undefined; 327 | } 328 | 329 | private detectTap(): void { 330 | if (this.eventType != undefined) { 331 | return; 332 | } 333 | 334 | const currentTime = new Date().getTime(); 335 | const tapLength = currentTime - this.touchstartTime; 336 | 337 | if (tapLength > 0) { 338 | if (tapLength < this.tapMinTimeout) { 339 | this.runHandler('tap', {}); 340 | } else { 341 | this.runHandler('longtap', {}); 342 | } 343 | } 344 | } 345 | 346 | private detectPinch(event: TouchEvent): boolean { 347 | const touches = event.touches; 348 | return (touches.length === 2 && this.eventType === undefined) || this.eventType === 'pinch'; 349 | } 350 | 351 | private detectLinearSwipe(event: MouseEvent | TouchEvent): 'vertical-swipe' | 'horizontal-swipe' { 352 | const touches = (event as TouchEvent).touches; 353 | 354 | if (touches) { 355 | if ( 356 | (touches.length === 1 && !this.eventType) || 357 | this.eventType === 'horizontal-swipe' || 358 | this.eventType === 'vertical-swipe' 359 | ) { 360 | return this.getLinearSwipeType(event); 361 | } 362 | } else { 363 | if (!this.eventType || this.eventType === 'horizontal-swipe' || this.eventType === 'vertical-swipe') { 364 | return this.getLinearSwipeType(event); 365 | } 366 | } 367 | 368 | return undefined; 369 | } 370 | 371 | private getLinearSwipeType(event: TouchEvent | MouseEvent): 'vertical-swipe' | 'horizontal-swipe' { 372 | if (this.eventType !== 'horizontal-swipe' && this.eventType !== 'vertical-swipe') { 373 | const movementX = Math.abs(this.moveLeft(0, event) - this.startX); 374 | const movementY = Math.abs(this.moveTop(0, event) - this.startY); 375 | 376 | if (movementY * 3 > movementX) { 377 | return 'vertical-swipe'; 378 | } else { 379 | return 'horizontal-swipe'; 380 | } 381 | } else { 382 | return this.eventType; 383 | } 384 | } 385 | 386 | private getElementPosition(): DOMRect { 387 | return this.element.getBoundingClientRect(); 388 | } 389 | 390 | private getTouchstartPosition(event: TouchEvent): void { 391 | this.startX = event.touches[0].clientX - this.elementPosition.left; 392 | this.startY = event.touches[0].clientY - this.elementPosition.top; 393 | } 394 | 395 | private getMousedownPosition(event: MouseEvent): void { 396 | this.startX = event.clientX - this.elementPosition.left; 397 | this.startY = event.clientY - this.elementPosition.top; 398 | } 399 | 400 | private moveLeft(index: number, event: TouchEvent | MouseEvent): number { 401 | const touches = (event as TouchEvent).touches; 402 | 403 | if (touches) { 404 | return touches[index].clientX - this.elementPosition.left; 405 | } else { 406 | return (event as MouseEvent).clientX - this.elementPosition.left; 407 | } 408 | } 409 | 410 | private moveTop(index: number, event: TouchEvent | MouseEvent): number { 411 | const touches = (event as TouchEvent).touches; 412 | 413 | if (touches) { 414 | return touches[index].clientY - this.elementPosition.top; 415 | } else { 416 | return (event as MouseEvent).clientY - this.elementPosition.top; 417 | } 418 | } 419 | 420 | private detectTouchScreen(): boolean { 421 | const prefixes = ' -webkit- -moz- -o- -ms- '.split(' '); 422 | const mq = (query: string): boolean => { 423 | return window.matchMedia(query).matches; 424 | }; 425 | 426 | if ('ontouchstart' in window) { 427 | return true; 428 | } 429 | 430 | // include the 'heartz' as a way to have a non matching MQ to help terminate the join 431 | // https://git.io/vznFH 432 | const query = ['(', prefixes.join('touch-enabled),('), 'heartz', ')'].join(''); 433 | return mq(query); 434 | } 435 | 436 | /* Public properties and methods */ 437 | public on(event: EventType, handler: (event: Event) => void): void { 438 | if (event) { 439 | this.handlers[event] = handler; 440 | } 441 | } 442 | } 443 | -------------------------------------------------------------------------------- /projects/ngx-pinch-zoom/src/lib/ivypinch.ts: -------------------------------------------------------------------------------- 1 | import { EventType, Touches } from './touches'; 2 | import { Properties } from './interfaces'; 3 | import { defaultProperties } from './properties'; 4 | import {EventEmitter} from "@angular/core"; 5 | 6 | export class IvyPinch { 7 | private readonly properties: Properties = defaultProperties; 8 | private touches: Touches; 9 | private readonly element: HTMLElement; 10 | private readonly elementTarget: string; 11 | private parentElement: HTMLElement; 12 | public scale: number = 1; 13 | private initialScale: number = 1; 14 | private elementPosition: DOMRect; 15 | private eventType: EventType; 16 | private startX: number = 0; 17 | private startY: number = 0; 18 | private moveX: number = 0; 19 | private moveY: number = 0; 20 | private initialMoveX: number = 0; 21 | private initialMoveY: number = 0; 22 | private moveXC: number = 0; 23 | private moveYC: number = 0; 24 | private distance: number = 0; 25 | private initialDistance: number = 0; 26 | public maxScale!: number; 27 | private defaultMaxScale: number = 3; 28 | private initialPinchCenterX = 0; 29 | private initialPinchCenterY = 0; 30 | private zoomChanged: EventEmitter; 31 | 32 | // Minimum scale at which panning works 33 | get minPanScale(): number { 34 | return this.getPropertiesValue('minPanScale'); 35 | } 36 | 37 | get fullImage(): { path: string; minScale?: number } { 38 | return this.properties.fullImage; 39 | } 40 | 41 | constructor(properties: Properties, zoomChanged: EventEmitter) { 42 | this.element = properties.element; 43 | this.zoomChanged = zoomChanged; 44 | 45 | if (!this.element) { 46 | return; 47 | } 48 | 49 | if (typeof properties.limitZoom === 'number') { 50 | this.maxScale = properties.limitZoom; 51 | } 52 | this.elementTarget = this.element.querySelector('*').tagName; 53 | this.parentElement = this.element.parentElement; 54 | this.properties = Object.assign({}, defaultProperties, properties); 55 | this.detectLimitZoom(); 56 | 57 | this.touches = new Touches({ 58 | element: properties.element, 59 | listeners: properties.listeners, 60 | resize: properties.autoHeight, 61 | mouseListeners: { 62 | mousedown: 'handleMousedown', 63 | mouseup: 'handleMouseup', 64 | wheel: 'handleWheel', 65 | }, 66 | }); 67 | 68 | /* Init */ 69 | this.setBasicStyles(); 70 | 71 | /* 72 | * Listeners 73 | */ 74 | 75 | this.touches.on('touchstart', this.handleTouchstart); 76 | this.touches.on('touchend', this.handleTouchend); 77 | this.touches.on('mousedown', this.handleTouchstart); 78 | this.touches.on('mouseup', this.handleTouchend); 79 | this.touches.on('pan', this.handlePan); 80 | this.touches.on('mousemove', this.handlePan); 81 | this.touches.on('pinch', this.handlePinch); 82 | 83 | if (this.properties.wheel) { 84 | this.touches.on('wheel', this.handleWheel); 85 | } 86 | 87 | if (this.properties.doubleTap) { 88 | this.touches.on('double-tap', this.handleDoubleTap); 89 | } 90 | 91 | if (this.properties.autoHeight) { 92 | this.touches.on('resize', this.handleResize); 93 | } 94 | } 95 | 96 | /* Touchstart */ 97 | 98 | private handleTouchstart = (event: TouchEvent | MouseEvent): void => { 99 | this.touches.addEventListeners('mousemove'); 100 | this.getElementPosition(); 101 | 102 | if (this.eventType === undefined) { 103 | this.getTouchstartPosition(event); 104 | } 105 | }; 106 | 107 | /* Touchend */ 108 | 109 | private handleTouchend = (event: TouchEvent | MouseEvent): void => { 110 | /* touchend */ 111 | if (event.type === 'touchend') { 112 | const touches = (event as TouchEvent).touches; 113 | 114 | // Min scale 115 | if (this.scale < 1) { 116 | this.scale = 1; 117 | this.zoomChanged.emit(this.scale); 118 | } 119 | 120 | // Auto Zoom Out 121 | if (this.properties.autoZoomOut && this.eventType === 'pinch') { 122 | this.scale = 1; 123 | this.zoomChanged.emit(this.scale); 124 | } 125 | 126 | // Align image 127 | if (this.eventType === 'pinch' || (this.eventType === 'pan' && this.scale > this.minPanScale)) { 128 | this.alignImage(); 129 | } 130 | 131 | // Update initial values 132 | if ( 133 | this.eventType === 'pinch' || 134 | this.eventType === 'pan' || 135 | this.eventType === 'horizontal-swipe' || 136 | this.eventType === 'vertical-swipe' 137 | ) { 138 | this.updateInitialValues(); 139 | } 140 | 141 | this.eventType = 'touchend'; 142 | 143 | if (touches && touches.length === 0) { 144 | this.eventType = undefined; 145 | } 146 | } 147 | 148 | /* mouseup */ 149 | if (event.type === 'mouseup') { 150 | this.updateInitialValues(); 151 | this.eventType = undefined; 152 | } 153 | 154 | this.touches.removeEventListeners('mousemove'); 155 | }; 156 | 157 | /* 158 | * Handlers 159 | */ 160 | 161 | private handlePan = (event: TouchEvent | MouseEvent): void => { 162 | if (this.scale < this.minPanScale || this.properties.disablePan) { 163 | return; 164 | } 165 | 166 | event.preventDefault(); 167 | const { clientX, clientY } = this.getClientPosition(event); 168 | 169 | if (!this.eventType) { 170 | this.startX = clientX - this.elementPosition.left; 171 | this.startY = clientY - this.elementPosition.top; 172 | } 173 | 174 | this.eventType = 'pan'; 175 | this.moveX = this.initialMoveX + (this.moveLeft(event, 0) - this.startX); 176 | this.moveY = this.initialMoveY + (this.moveTop(event, 0) - this.startY); 177 | 178 | if (this.properties.limitPan) { 179 | this.limitPanY(); 180 | this.limitPanX(); 181 | } 182 | 183 | /* mousemove */ 184 | if (event.type === 'mousemove' && this.scale > this.minPanScale) { 185 | this.centeringImage(); 186 | } 187 | 188 | this.transformElement(0); 189 | }; 190 | 191 | private handleDoubleTap = (event: TouchEvent): void => { 192 | this.toggleZoom(event); 193 | return; 194 | }; 195 | 196 | private handlePinch = (event: TouchEvent): void => { 197 | event.preventDefault(); 198 | 199 | if (!this.properties.draggableOnPinch) { 200 | if (this.eventType === undefined || this.eventType === 'pinch') { 201 | const touches = event.touches; 202 | 203 | if (!this.eventType) { 204 | this.initialDistance = this.getDistance(touches); 205 | 206 | const moveLeft0 = this.moveLeft(event, 0); 207 | const moveLeft1 = this.moveLeft(event, 1); 208 | const moveTop0 = this.moveTop(event, 0); 209 | const moveTop1 = this.moveTop(event, 1); 210 | 211 | this.moveXC = (moveLeft0 + moveLeft1) / 2 - this.initialMoveX; 212 | this.moveYC = (moveTop0 + moveTop1) / 2 - this.initialMoveY; 213 | } 214 | 215 | this.eventType = 'pinch'; 216 | this.distance = this.getDistance(touches); 217 | this.scale = this.initialScale * (this.distance / this.initialDistance); 218 | this.zoomChanged.emit(this.scale); 219 | this.moveX = this.initialMoveX - ((this.distance / this.initialDistance) * this.moveXC - this.moveXC); 220 | this.moveY = this.initialMoveY - ((this.distance / this.initialDistance) * this.moveYC - this.moveYC); 221 | 222 | this.handleLimitZoom(); 223 | 224 | if (this.properties.limitPan) { 225 | this.limitPanY(); 226 | this.limitPanX(); 227 | } 228 | 229 | this.transformElement(0); 230 | } 231 | 232 | return; 233 | } 234 | 235 | const touches = event.touches; 236 | 237 | if (!this.eventType) { 238 | this.eventType = 'pinch'; 239 | this.initialDistance = this.getDistance(touches); 240 | const lx0 = this.moveLeft(event, 0), 241 | lx1 = this.moveLeft(event, 1), 242 | ty0 = this.moveTop(event, 0), 243 | ty1 = this.moveTop(event, 1); 244 | this.initialPinchCenterX = (lx0 + lx1) / 2; 245 | this.initialPinchCenterY = (ty0 + ty1) / 2; 246 | this.moveXC = this.initialPinchCenterX - this.initialMoveX; 247 | this.moveYC = this.initialPinchCenterY - this.initialMoveY; 248 | } 249 | 250 | this.eventType = 'pinch'; 251 | this.distance = this.getDistance(touches); 252 | const scaleRatio = this.distance / this.initialDistance; 253 | this.scale = this.initialScale * scaleRatio; 254 | this.zoomChanged.emit(this.scale); 255 | 256 | const curLX0 = this.moveLeft(event, 0), 257 | curLX1 = this.moveLeft(event, 1), 258 | curTY0 = this.moveTop(event, 0), 259 | curTY1 = this.moveTop(event, 1); 260 | const currentCenterX = (curLX0 + curLX1) / 2; 261 | const currentCenterY = (curTY0 + curTY1) / 2; 262 | 263 | const deltaX = currentCenterX - this.initialPinchCenterX; 264 | const deltaY = currentCenterY - this.initialPinchCenterY; 265 | const scaleTransX = (scaleRatio - 1) * this.moveXC; 266 | const scaleTransY = (scaleRatio - 1) * this.moveYC; 267 | 268 | this.moveX = this.initialMoveX + deltaX - scaleTransX; 269 | this.moveY = this.initialMoveY + deltaY - scaleTransY; 270 | 271 | this.handleLimitZoom(); 272 | if (this.properties.limitPan) { 273 | this.limitPanY(); 274 | this.limitPanX(); 275 | } 276 | this.transformElement(0); 277 | }; 278 | 279 | private handleWheel = (event: WheelEvent): void => { 280 | event.preventDefault(); 281 | 282 | const wheelZoomFactor = this.properties.wheelZoomFactor || 0; 283 | const zoomFactor = event.deltaY < 0 ? wheelZoomFactor : -wheelZoomFactor; 284 | let newScale = this.initialScale + zoomFactor; 285 | 286 | /* Round value */ 287 | if (newScale < 1 + wheelZoomFactor) { 288 | newScale = 1; 289 | } else if (newScale < this.maxScale && newScale > this.maxScale - wheelZoomFactor) { 290 | newScale = this.maxScale; 291 | } 292 | 293 | if (newScale < 1 || newScale > this.maxScale) { 294 | return; 295 | } 296 | 297 | if (newScale === this.scale) { 298 | return; 299 | } 300 | 301 | this.getElementPosition(); 302 | this.scale = newScale; 303 | this.zoomChanged.emit(this.scale); 304 | 305 | 306 | /* Get cursor position over image */ 307 | const xCenter = event.clientX - this.elementPosition.left - this.initialMoveX; 308 | const yCenter = event.clientY - this.elementPosition.top - this.initialMoveY; 309 | 310 | this.setZoom({ 311 | scale: newScale, 312 | center: [xCenter, yCenter], 313 | }); 314 | }; 315 | 316 | private handleResize = (_event: Event): void => { 317 | this.setAutoHeight(); 318 | }; 319 | 320 | private handleLimitZoom(): void { 321 | const limitZoom = this.maxScale; 322 | const minScale = this.properties.minScale || 0; 323 | 324 | if (this.scale > limitZoom || this.scale <= minScale) { 325 | const imageWidth = this.getImageWidth(); 326 | const imageHeight = this.getImageHeight(); 327 | const enlargedImageWidth = imageWidth * this.scale; 328 | const enlargedImageHeight = imageHeight * this.scale; 329 | const moveXRatio = this.moveX / (enlargedImageWidth - imageWidth); 330 | const moveYRatio = this.moveY / (enlargedImageHeight - imageHeight); 331 | 332 | if (this.scale > limitZoom) { 333 | this.scale = limitZoom; 334 | this.zoomChanged.emit(this.scale); 335 | } 336 | 337 | if (this.scale <= minScale) { 338 | this.scale = minScale; 339 | this.zoomChanged.emit(this.scale); 340 | } 341 | 342 | const newImageWidth = imageWidth * this.scale; 343 | const newImageHeight = imageHeight * this.scale; 344 | 345 | this.moveX = -Math.abs(moveXRatio * (newImageWidth - imageWidth)); 346 | this.moveY = -Math.abs(-moveYRatio * (newImageHeight - imageHeight)); 347 | } 348 | } 349 | 350 | private moveLeft(event: TouchEvent | MouseEvent, index: number = 0): number { 351 | const clientX = this.getClientPosition(event, index).clientX; 352 | return clientX - this.elementPosition.left; 353 | } 354 | 355 | private moveTop(event: TouchEvent | MouseEvent, index: number = 0): number { 356 | const clientY = this.getClientPosition(event, index).clientY; 357 | return clientY - this.elementPosition.top; 358 | } 359 | 360 | /* 361 | * Detection 362 | */ 363 | 364 | private centeringImage(): boolean { 365 | const img = this.getImageElement(); 366 | const initialMoveX = this.moveX; 367 | const initialMoveY = this.moveY; 368 | 369 | if (this.moveY > 0) { 370 | this.moveY = 0; 371 | } 372 | if (this.moveX > 0) { 373 | this.moveX = 0; 374 | } 375 | 376 | if (img) { 377 | this.limitPanY(); 378 | this.limitPanX(); 379 | } 380 | if (img && this.scale < 1) { 381 | if (this.moveX < this.element.offsetWidth * (1 - this.scale)) { 382 | this.moveX = this.element.offsetWidth * (1 - this.scale); 383 | } 384 | } 385 | 386 | return initialMoveX !== this.moveX || initialMoveY !== this.moveY; 387 | } 388 | 389 | private limitPanY(): void { 390 | const imgHeight = this.getImageHeight(); 391 | const scaledImgHeight = imgHeight * this.scale; 392 | const parentHeight = this.parentElement.offsetHeight; 393 | const elementHeight = this.element.offsetHeight; 394 | 395 | if (scaledImgHeight < parentHeight) { 396 | this.moveY = (parentHeight - elementHeight * this.scale) / 2; 397 | } else { 398 | const imgOffsetTop = ((imgHeight - elementHeight) * this.scale) / 2; 399 | 400 | if (this.moveY > imgOffsetTop) { 401 | this.moveY = imgOffsetTop; 402 | } else if (scaledImgHeight + Math.abs(imgOffsetTop) - parentHeight + this.moveY < 0) { 403 | this.moveY = -(scaledImgHeight + Math.abs(imgOffsetTop) - parentHeight); 404 | } 405 | } 406 | } 407 | 408 | private limitPanX(): void { 409 | const imgWidth = this.getImageWidth(); 410 | const scaledImgWidth = imgWidth * this.scale; 411 | const parentWidth = this.parentElement.offsetWidth; 412 | const elementWidth = this.element.offsetWidth; 413 | 414 | if (scaledImgWidth < parentWidth) { 415 | this.moveX = (parentWidth - elementWidth * this.scale) / 2; 416 | } else { 417 | const imgOffsetLeft = ((imgWidth - elementWidth) * this.scale) / 2; 418 | 419 | if (this.moveX > imgOffsetLeft) { 420 | this.moveX = imgOffsetLeft; 421 | } else if (scaledImgWidth + Math.abs(imgOffsetLeft) - parentWidth + this.moveX < 0) { 422 | this.moveX = -(imgWidth * this.scale + Math.abs(imgOffsetLeft) - parentWidth); 423 | } 424 | } 425 | } 426 | 427 | private setBasicStyles(): void { 428 | this.element.style.display = 'flex'; 429 | this.element.style.alignItems = 'center'; 430 | this.element.style.justifyContent = 'center'; 431 | this.element.style.transformOrigin = '0 0'; 432 | this.setImageSize(); 433 | this.setDraggableImage(); 434 | } 435 | 436 | private removeBasicStyles(): void { 437 | this.element.style.display = ''; 438 | this.element.style.alignItems = ''; 439 | this.element.style.justifyContent = ''; 440 | this.element.style.transformOrigin = ''; 441 | this.removeImageSize(); 442 | this.removeDraggableImage(); 443 | } 444 | 445 | private setDraggableImage(): void { 446 | const imgElement = this.getImageElement(); 447 | 448 | if (imgElement) { 449 | imgElement.draggable = this.properties.draggableImage; 450 | } 451 | } 452 | 453 | private removeDraggableImage(): void { 454 | const imgElement = this.getImageElement(); 455 | 456 | if (imgElement) { 457 | imgElement.draggable = true; 458 | } 459 | } 460 | 461 | private setImageSize(): void { 462 | const imgElement = this.getImageElements(); 463 | 464 | if (imgElement.length) { 465 | imgElement[0].style.maxWidth = '100%'; 466 | imgElement[0].style.maxHeight = '100%'; 467 | 468 | this.setAutoHeight(); 469 | } 470 | } 471 | 472 | private setAutoHeight(): void { 473 | const imgElement = this.getImageElements(); 474 | 475 | if (!this.properties.autoHeight || !imgElement.length) { 476 | return; 477 | } 478 | 479 | const imgNaturalWidth = imgElement[0].getAttribute('width'); 480 | const imgNaturalHeight = imgElement[0].getAttribute('height'); 481 | const sizeRatio = +imgNaturalWidth / +imgNaturalHeight; 482 | const parentWidth = this.parentElement.offsetWidth; 483 | 484 | imgElement[0].style.maxHeight = parentWidth / sizeRatio + 'px'; 485 | } 486 | 487 | private removeImageSize(): void { 488 | const imgElement = this.getImageElements(); 489 | 490 | if (imgElement.length) { 491 | imgElement[0].style.maxWidth = ''; 492 | imgElement[0].style.maxHeight = ''; 493 | } 494 | } 495 | 496 | private getElementPosition(): void { 497 | this.elementPosition = this.element.parentElement.getBoundingClientRect(); 498 | } 499 | 500 | private getTouchstartPosition(event: TouchEvent | MouseEvent): void { 501 | const { clientX, clientY } = this.getClientPosition(event); 502 | 503 | this.startX = clientX - this.elementPosition.left; 504 | this.startY = clientY - this.elementPosition.top; 505 | } 506 | 507 | private getClientPosition(event: TouchEvent | MouseEvent, index: number = 0): { clientX: number; clientY: number } { 508 | let clientX: number; 509 | let clientY: number; 510 | 511 | if (event.type === 'touchstart' || event.type === 'touchmove') { 512 | clientX = (event as TouchEvent).touches[index].clientX; 513 | clientY = (event as TouchEvent).touches[index].clientY; 514 | } 515 | if (event.type === 'mousedown' || event.type === 'mousemove') { 516 | clientX = (event as MouseEvent).clientX; 517 | clientY = (event as MouseEvent).clientY; 518 | } 519 | 520 | return { 521 | clientX, 522 | clientY, 523 | }; 524 | } 525 | 526 | private resetScale(): void { 527 | this.scale = 1; 528 | this.zoomChanged.emit(this.scale); 529 | this.moveX = 0; 530 | this.moveY = 0; 531 | this.updateInitialValues(); 532 | this.transformElement(this.properties.transitionDuration); 533 | } 534 | 535 | private updateInitialValues(): void { 536 | this.initialScale = this.scale; 537 | this.initialMoveX = this.moveX; 538 | this.initialMoveY = this.moveY; 539 | } 540 | 541 | private getDistance(touches: TouchList): number { 542 | return Math.sqrt( 543 | Math.pow(touches[0].pageX - touches[1].pageX, 2) + Math.pow(touches[0].pageY - touches[1].pageY, 2), 544 | ); 545 | } 546 | 547 | private getImageHeight(): number { 548 | const img = this.getImageElement() as HTMLImageElement; 549 | return img.offsetHeight; 550 | } 551 | 552 | private getImageWidth(): number { 553 | const img = this.getImageElement() as HTMLImageElement; 554 | return img.offsetWidth; 555 | } 556 | 557 | private transformElement(duration: number): void { 558 | this.element.style.transition = 'all ' + duration + 'ms'; 559 | this.element.style.transform = 560 | 'matrix(' + 561 | Number(this.scale) + 562 | ', 0, 0, ' + 563 | Number(this.scale) + 564 | ', ' + 565 | Number(this.moveX) + 566 | ', ' + 567 | Number(this.moveY) + 568 | ')'; 569 | } 570 | 571 | private isTouchScreen(): boolean { 572 | const prefixes = ' -webkit- -moz- -o- -ms- '.split(' '); 573 | 574 | if ('ontouchstart' in window) { 575 | return true; 576 | } 577 | 578 | // include the 'heartz' as a way to have a non matching MQ to help terminate the join 579 | // https://git.io/vznFH 580 | const query = ['(', prefixes.join('touch-enabled),('), 'heartz', ')'].join(''); 581 | return this.getMatchMedia(query); 582 | } 583 | 584 | private getMatchMedia(query: string): boolean { 585 | return window.matchMedia(query).matches; 586 | } 587 | 588 | public isDragging(): boolean { 589 | if (this.properties.disablePan) { 590 | return false; 591 | } 592 | 593 | const imgHeight = this.getImageHeight(); 594 | const imgWidth = this.getImageWidth(); 595 | 596 | if (this.scale > 1) { 597 | return ( 598 | imgHeight * this.scale > this.parentElement.offsetHeight || 599 | imgWidth * this.scale > this.parentElement.offsetWidth 600 | ); 601 | } 602 | if (this.scale === 1) { 603 | return imgHeight > this.parentElement.offsetHeight || imgWidth > this.parentElement.offsetWidth; 604 | } 605 | 606 | return undefined; 607 | } 608 | 609 | public detectLimitZoom(): void { 610 | // Assign to default only if it is not passed through constructor 611 | this.maxScale ??= this.defaultMaxScale; 612 | 613 | if (this.properties.limitZoom === 'original image size' && this.elementTarget === 'IMG') { 614 | // We are waiting for the element with the image to be available 615 | this.pollLimitZoomForOriginalImage(); 616 | } 617 | } 618 | 619 | private pollLimitZoomForOriginalImage(): void { 620 | const poll = setInterval(() => { 621 | const maxScaleForOriginalImage = this.getMaxScaleForOriginalImage(); 622 | if (typeof maxScaleForOriginalImage === 'number') { 623 | this.maxScale = maxScaleForOriginalImage; 624 | clearInterval(poll); 625 | } 626 | }, 10); 627 | } 628 | 629 | private getMaxScaleForOriginalImage(): number { 630 | let maxScale!: number; 631 | const img = this.element.getElementsByTagName('img')[0]; 632 | 633 | if (img.naturalWidth && img.offsetWidth) { 634 | maxScale = img.naturalWidth / img.offsetWidth; 635 | } 636 | 637 | return maxScale; 638 | } 639 | 640 | private getImageElement(): HTMLElement { 641 | const imgElement = this.element.getElementsByTagName(this.elementTarget); 642 | 643 | if (imgElement.length) { 644 | return imgElement[0] as HTMLElement; 645 | } 646 | } 647 | 648 | private getImageElements(): HTMLCollectionOf { 649 | return this.element.getElementsByTagName(this.elementTarget) as HTMLCollectionOf; 650 | } 651 | 652 | public toggleZoom(event: TouchEvent | boolean = false): void { 653 | if (this.initialScale === 1) { 654 | if (event && (event as TouchEvent).changedTouches) { 655 | if (this.properties.doubleTapScale === undefined) { 656 | return; 657 | } 658 | 659 | const changedTouches = (event as TouchEvent).changedTouches; 660 | this.scale = this.initialScale * this.properties.doubleTapScale; 661 | this.zoomChanged.emit(this.scale); 662 | this.moveX = 663 | this.initialMoveX - 664 | (changedTouches[0].clientX - this.elementPosition.left) * (this.properties.doubleTapScale - 1); 665 | this.moveY = 666 | this.initialMoveY - 667 | (changedTouches[0].clientY - this.elementPosition.top) * (this.properties.doubleTapScale - 1); 668 | } else { 669 | const zoomControlScale = this.properties.zoomControlScale || 0; 670 | this.scale = this.initialScale * (zoomControlScale + 1); 671 | this.zoomChanged.emit(this.scale); 672 | this.moveX = this.initialMoveX - (this.element.offsetWidth * (this.scale - 1)) / 2; 673 | this.moveY = this.initialMoveY - (this.element.offsetHeight * (this.scale - 1)) / 2; 674 | } 675 | 676 | this.centeringImage(); 677 | this.updateInitialValues(); 678 | this.transformElement(this.properties.transitionDuration); 679 | } else { 680 | this.resetScale(); 681 | } 682 | } 683 | 684 | public zoomIn(value: number): number { 685 | const scale = this.scale + value; 686 | 687 | if (scale >= this.maxScale) { 688 | this.scale = this.maxScale; 689 | this.zoomChanged.emit(this.scale); 690 | 691 | return this.scale; 692 | } 693 | 694 | this.setZoom({ scale }); 695 | 696 | return this.scale; 697 | } 698 | 699 | public zoomOut(value: number): number { 700 | const scale = this.scale - value; 701 | 702 | if (scale <= this.properties.minScale) { 703 | this.scale = this.properties.minScale; 704 | this.zoomChanged.emit(this.scale); 705 | 706 | return this.scale; 707 | } 708 | 709 | this.setZoom({ scale }); 710 | 711 | return this.scale; 712 | } 713 | 714 | private setZoom(properties: { scale: number; center?: number[] }): void { 715 | this.scale = properties.scale; 716 | this.zoomChanged.emit(this.scale); 717 | let xCenter; 718 | let yCenter; 719 | const visibleAreaWidth = this.element.offsetWidth; 720 | const visibleAreaHeight = this.element.offsetHeight; 721 | const scalingPercent = (visibleAreaWidth * this.scale) / (visibleAreaWidth * this.initialScale); 722 | 723 | if (properties.center) { 724 | xCenter = properties.center[0]; 725 | yCenter = properties.center[1]; 726 | } else { 727 | xCenter = visibleAreaWidth / 2 - this.initialMoveX; 728 | yCenter = visibleAreaHeight / 2 - this.initialMoveY; 729 | } 730 | 731 | this.moveX = this.initialMoveX - (scalingPercent * xCenter - xCenter); 732 | this.moveY = this.initialMoveY - (scalingPercent * yCenter - yCenter); 733 | 734 | this.centeringImage(); 735 | this.updateInitialValues(); 736 | this.transformElement(this.properties.transitionDuration); 737 | } 738 | 739 | private alignImage(): void { 740 | const isMoveChanged = this.centeringImage(); 741 | 742 | if (isMoveChanged) { 743 | this.updateInitialValues(); 744 | this.transformElement(this.properties.transitionDuration); 745 | } 746 | } 747 | 748 | public destroy(): void { 749 | this.removeBasicStyles(); 750 | this.touches.destroy(); 751 | } 752 | 753 | private getPropertiesValue(propertyName: K): Properties[K] { 754 | if (this.properties && this.properties[propertyName]) { 755 | return this.properties[propertyName]; 756 | } else { 757 | return defaultProperties[propertyName]; 758 | } 759 | } 760 | } 761 | --------------------------------------------------------------------------------