ngx-image-cropper 5.0.0 → 6.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (64) hide show
  1. package/README.md +1 -0
  2. package/esm2020/lib/component/image-cropper.component.mjs +540 -0
  3. package/esm2020/lib/image-cropper.module.mjs +26 -0
  4. package/esm2020/lib/interfaces/cropper-options.interface.mjs +2 -0
  5. package/esm2020/lib/interfaces/cropper-position.interface.mjs +2 -0
  6. package/esm2020/lib/interfaces/cropper.settings.mjs +50 -0
  7. package/esm2020/lib/interfaces/dimensions.interface.mjs +2 -0
  8. package/esm2020/lib/interfaces/exif-transform.interface.mjs +2 -0
  9. package/esm2020/lib/interfaces/image-cropped-event.interface.mjs +2 -0
  10. package/esm2020/lib/interfaces/image-transform.interface.mjs +2 -0
  11. package/esm2020/lib/interfaces/index.mjs +2 -0
  12. package/esm2020/lib/interfaces/loaded-image.interface.mjs +2 -0
  13. package/esm2020/lib/interfaces/move-start.interface.mjs +7 -0
  14. package/esm2020/lib/services/crop.service.mjs +117 -0
  15. package/esm2020/lib/services/cropper-position.service.mjs +201 -0
  16. package/esm2020/lib/services/load-image.service.mjs +168 -0
  17. package/esm2020/lib/utils/blob.utils.mjs +12 -0
  18. package/esm2020/lib/utils/exif.utils.mjs +89 -0
  19. package/esm2020/lib/utils/hammer.utils.mjs +2 -0
  20. package/esm2020/lib/utils/keyboard.utils.mjs +40 -0
  21. package/esm2020/lib/utils/resize.utils.mjs +75 -0
  22. package/esm2020/ngx-image-cropper.mjs +5 -0
  23. package/esm2020/public-api.mjs +6 -0
  24. package/fesm2015/{ngx-image-cropper.js → ngx-image-cropper.mjs} +251 -1035
  25. package/fesm2015/ngx-image-cropper.mjs.map +1 -0
  26. package/fesm2020/ngx-image-cropper.mjs +1315 -0
  27. package/fesm2020/ngx-image-cropper.mjs.map +1 -0
  28. package/lib/component/image-cropper.component.d.ts +12 -9
  29. package/lib/image-cropper.module.d.ts +6 -0
  30. package/lib/interfaces/cropper.settings.d.ts +1 -1
  31. package/lib/services/crop.service.d.ts +3 -0
  32. package/lib/services/cropper-position.service.d.ts +3 -0
  33. package/lib/services/load-image.service.d.ts +3 -0
  34. package/lib/utils/hammer.utils.d.ts +1 -1
  35. package/ngx-image-cropper.d.ts +1 -3
  36. package/package.json +24 -13
  37. package/bundles/ngx-image-cropper.umd.js +0 -2418
  38. package/bundles/ngx-image-cropper.umd.js.map +0 -1
  39. package/bundles/ngx-image-cropper.umd.min.js +0 -16
  40. package/bundles/ngx-image-cropper.umd.min.js.map +0 -1
  41. package/esm2015/lib/component/image-cropper.component.js +0 -827
  42. package/esm2015/lib/image-cropper.module.js +0 -24
  43. package/esm2015/lib/interfaces/cropper-options.interface.js +0 -56
  44. package/esm2015/lib/interfaces/cropper-position.interface.js +0 -20
  45. package/esm2015/lib/interfaces/cropper.settings.js +0 -140
  46. package/esm2015/lib/interfaces/dimensions.interface.js +0 -16
  47. package/esm2015/lib/interfaces/exif-transform.interface.js +0 -16
  48. package/esm2015/lib/interfaces/image-cropped-event.interface.js +0 -24
  49. package/esm2015/lib/interfaces/image-transform.interface.js +0 -20
  50. package/esm2015/lib/interfaces/index.js +0 -6
  51. package/esm2015/lib/interfaces/loaded-image.interface.js +0 -18
  52. package/esm2015/lib/interfaces/move-start.interface.js +0 -37
  53. package/esm2015/lib/services/crop.service.js +0 -176
  54. package/esm2015/lib/services/cropper-position.service.js +0 -259
  55. package/esm2015/lib/services/load-image.service.js +0 -304
  56. package/esm2015/lib/utils/blob.utils.js +0 -26
  57. package/esm2015/lib/utils/exif.utils.js +0 -128
  58. package/esm2015/lib/utils/hammer.utils.js +0 -29
  59. package/esm2015/lib/utils/keyboard.utils.js +0 -58
  60. package/esm2015/lib/utils/resize.utils.js +0 -116
  61. package/esm2015/ngx-image-cropper.js +0 -13
  62. package/esm2015/public-api.js +0 -11
  63. package/fesm2015/ngx-image-cropper.js.map +0 -1
  64. package/ngx-image-cropper.metadata.json +0 -1
@@ -1,304 +0,0 @@
1
- /**
2
- * @fileoverview added by tsickle
3
- * Generated from: lib/services/load-image.service.ts
4
- * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
- */
6
- import { __awaiter } from "tslib";
7
- import { Injectable } from '@angular/core';
8
- import { getTransformationsFromExifData, supportsAutomaticRotation } from '../utils/exif.utils';
9
- import * as i0 from "@angular/core";
10
- /**
11
- * @record
12
- */
13
- function LoadImageBase64() { }
14
- if (false) {
15
- /** @type {?} */
16
- LoadImageBase64.prototype.originalImage;
17
- /** @type {?} */
18
- LoadImageBase64.prototype.originalBase64;
19
- }
20
- export class LoadImageService {
21
- constructor() {
22
- this.autoRotateSupported = supportsAutomaticRotation();
23
- }
24
- /**
25
- * @param {?} file
26
- * @param {?} cropperSettings
27
- * @return {?}
28
- */
29
- loadImageFile(file, cropperSettings) {
30
- return new Promise((/**
31
- * @param {?} resolve
32
- * @param {?} reject
33
- * @return {?}
34
- */
35
- (resolve, reject) => {
36
- /** @type {?} */
37
- const fileReader = new FileReader();
38
- fileReader.onload = (/**
39
- * @param {?} event
40
- * @return {?}
41
- */
42
- (event) => {
43
- this.loadImage(event.target.result, file.type, cropperSettings)
44
- .then(resolve)
45
- .catch(reject);
46
- });
47
- fileReader.readAsDataURL(file);
48
- }));
49
- }
50
- /**
51
- * @private
52
- * @param {?} imageBase64
53
- * @param {?} imageType
54
- * @param {?} cropperSettings
55
- * @return {?}
56
- */
57
- loadImage(imageBase64, imageType, cropperSettings) {
58
- if (!this.isValidImageType(imageType)) {
59
- return Promise.reject(new Error('Invalid image type'));
60
- }
61
- return this.loadBase64Image(imageBase64, cropperSettings);
62
- }
63
- /**
64
- * @private
65
- * @param {?} type
66
- * @return {?}
67
- */
68
- isValidImageType(type) {
69
- return /image\/(png|jpg|jpeg|bmp|gif|tiff|webp)/.test(type);
70
- }
71
- /**
72
- * @param {?} url
73
- * @param {?} cropperSettings
74
- * @return {?}
75
- */
76
- loadImageFromURL(url, cropperSettings) {
77
- return new Promise((/**
78
- * @param {?} resolve
79
- * @param {?} reject
80
- * @return {?}
81
- */
82
- (resolve, reject) => {
83
- /** @type {?} */
84
- const img = new Image();
85
- img.onerror = (/**
86
- * @return {?}
87
- */
88
- () => reject);
89
- img.onload = (/**
90
- * @return {?}
91
- */
92
- () => {
93
- /** @type {?} */
94
- const canvas = document.createElement('canvas');
95
- /** @type {?} */
96
- const context = canvas.getContext('2d');
97
- canvas.width = img.width;
98
- canvas.height = img.height;
99
- context.drawImage(img, 0, 0);
100
- this.loadBase64Image(canvas.toDataURL(), cropperSettings).then(resolve);
101
- });
102
- img.crossOrigin = 'anonymous';
103
- img.src = url;
104
- }));
105
- }
106
- /**
107
- * @param {?} imageBase64
108
- * @param {?} cropperSettings
109
- * @return {?}
110
- */
111
- loadBase64Image(imageBase64, cropperSettings) {
112
- return new Promise((/**
113
- * @param {?} resolve
114
- * @param {?} reject
115
- * @return {?}
116
- */
117
- (resolve, reject) => {
118
- /** @type {?} */
119
- const originalImage = new Image();
120
- originalImage.onload = (/**
121
- * @return {?}
122
- */
123
- () => resolve({
124
- originalImage,
125
- originalBase64: imageBase64
126
- }));
127
- originalImage.onerror = reject;
128
- originalImage.src = imageBase64;
129
- })).then((/**
130
- * @param {?} res
131
- * @return {?}
132
- */
133
- (res) => this.transformImageBase64(res, cropperSettings)));
134
- }
135
- /**
136
- * @private
137
- * @param {?} res
138
- * @param {?} cropperSettings
139
- * @return {?}
140
- */
141
- transformImageBase64(res, cropperSettings) {
142
- return __awaiter(this, void 0, void 0, function* () {
143
- /** @type {?} */
144
- const autoRotate = yield this.autoRotateSupported;
145
- /** @type {?} */
146
- const exifTransform = yield getTransformationsFromExifData(autoRotate ? -1 : res.originalBase64);
147
- if (!res.originalImage || !res.originalImage.complete) {
148
- return Promise.reject(new Error('No image loaded'));
149
- }
150
- /** @type {?} */
151
- const loadedImage = {
152
- original: {
153
- base64: res.originalBase64,
154
- image: res.originalImage,
155
- size: {
156
- width: res.originalImage.naturalWidth,
157
- height: res.originalImage.naturalHeight
158
- }
159
- },
160
- exifTransform
161
- };
162
- return this.transformLoadedImage(loadedImage, cropperSettings);
163
- });
164
- }
165
- /**
166
- * @param {?} loadedImage
167
- * @param {?} cropperSettings
168
- * @return {?}
169
- */
170
- transformLoadedImage(loadedImage, cropperSettings) {
171
- return __awaiter(this, void 0, void 0, function* () {
172
- /** @type {?} */
173
- const canvasRotation = cropperSettings.canvasRotation + loadedImage.exifTransform.rotate;
174
- /** @type {?} */
175
- const originalSize = {
176
- width: loadedImage.original.image.naturalWidth,
177
- height: loadedImage.original.image.naturalHeight
178
- };
179
- if (canvasRotation === 0 && !loadedImage.exifTransform.flip && !cropperSettings.containWithinAspectRatio) {
180
- return {
181
- original: {
182
- base64: loadedImage.original.base64,
183
- image: loadedImage.original.image,
184
- size: Object.assign({}, originalSize)
185
- },
186
- transformed: {
187
- base64: loadedImage.original.base64,
188
- image: loadedImage.original.image,
189
- size: Object.assign({}, originalSize)
190
- },
191
- exifTransform: loadedImage.exifTransform
192
- };
193
- }
194
- /** @type {?} */
195
- const transformedSize = this.getTransformedSize(originalSize, loadedImage.exifTransform, cropperSettings);
196
- /** @type {?} */
197
- const canvas = document.createElement('canvas');
198
- canvas.width = transformedSize.width;
199
- canvas.height = transformedSize.height;
200
- /** @type {?} */
201
- const ctx = canvas.getContext('2d');
202
- ctx.setTransform(loadedImage.exifTransform.flip ? -1 : 1, 0, 0, 1, canvas.width / 2, canvas.height / 2);
203
- ctx.rotate(Math.PI * (canvasRotation / 2));
204
- ctx.drawImage(loadedImage.original.image, -originalSize.width / 2, -originalSize.height / 2);
205
- /** @type {?} */
206
- const transformedBase64 = canvas.toDataURL();
207
- /** @type {?} */
208
- const transformedImage = yield this.loadImageFromBase64(transformedBase64);
209
- return {
210
- original: {
211
- base64: loadedImage.original.base64,
212
- image: loadedImage.original.image,
213
- size: Object.assign({}, originalSize)
214
- },
215
- transformed: {
216
- base64: transformedBase64,
217
- image: transformedImage,
218
- size: {
219
- width: transformedImage.width,
220
- height: transformedImage.height
221
- }
222
- },
223
- exifTransform: loadedImage.exifTransform
224
- };
225
- });
226
- }
227
- /**
228
- * @private
229
- * @param {?} imageBase64
230
- * @return {?}
231
- */
232
- loadImageFromBase64(imageBase64) {
233
- return new Promise(((/**
234
- * @param {?} resolve
235
- * @param {?} reject
236
- * @return {?}
237
- */
238
- (resolve, reject) => {
239
- /** @type {?} */
240
- const image = new Image();
241
- image.onload = (/**
242
- * @return {?}
243
- */
244
- () => resolve(image));
245
- image.onerror = reject;
246
- image.src = imageBase64;
247
- })));
248
- }
249
- /**
250
- * @private
251
- * @param {?} originalSize
252
- * @param {?} exifTransform
253
- * @param {?} cropperSettings
254
- * @return {?}
255
- */
256
- getTransformedSize(originalSize, exifTransform, cropperSettings) {
257
- /** @type {?} */
258
- const canvasRotation = cropperSettings.canvasRotation + exifTransform.rotate;
259
- if (cropperSettings.containWithinAspectRatio) {
260
- if (canvasRotation % 2) {
261
- /** @type {?} */
262
- const minWidthToContain = originalSize.width * cropperSettings.aspectRatio;
263
- /** @type {?} */
264
- const minHeightToContain = originalSize.height / cropperSettings.aspectRatio;
265
- return {
266
- width: Math.max(originalSize.height, minWidthToContain),
267
- height: Math.max(originalSize.width, minHeightToContain)
268
- };
269
- }
270
- else {
271
- /** @type {?} */
272
- const minWidthToContain = originalSize.height * cropperSettings.aspectRatio;
273
- /** @type {?} */
274
- const minHeightToContain = originalSize.width / cropperSettings.aspectRatio;
275
- return {
276
- width: Math.max(originalSize.width, minWidthToContain),
277
- height: Math.max(originalSize.height, minHeightToContain)
278
- };
279
- }
280
- }
281
- if (canvasRotation % 2) {
282
- return {
283
- height: originalSize.width,
284
- width: originalSize.height
285
- };
286
- }
287
- return {
288
- width: originalSize.width,
289
- height: originalSize.height
290
- };
291
- }
292
- }
293
- LoadImageService.decorators = [
294
- { type: Injectable, args: [{ providedIn: 'root' },] }
295
- ];
296
- /** @nocollapse */ LoadImageService.ɵprov = i0.ɵɵdefineInjectable({ factory: function LoadImageService_Factory() { return new LoadImageService(); }, token: LoadImageService, providedIn: "root" });
297
- if (false) {
298
- /**
299
- * @type {?}
300
- * @private
301
- */
302
- LoadImageService.prototype.autoRotateSupported;
303
- }
304
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"load-image.service.js","sourceRoot":"/Users/mawi/Projects/ngx-image-cropper/projects/ngx-image-cropper/src/","sources":["lib/services/load-image.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Dimensions, LoadedImage } from '../interfaces';\nimport { CropperSettings } from '../interfaces/cropper.settings';\nimport { ExifTransform } from '../interfaces/exif-transform.interface';\nimport { getTransformationsFromExifData, supportsAutomaticRotation } from '../utils/exif.utils';\n\ninterface LoadImageBase64 {\n  originalImage: HTMLImageElement;\n  originalBase64: string;\n}\n\n@Injectable({providedIn: 'root'})\nexport class LoadImageService {\n\n  private autoRotateSupported: Promise<boolean> = supportsAutomaticRotation();\n\n  loadImageFile(file: File, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    return new Promise((resolve, reject) => {\n      const fileReader = new FileReader();\n      fileReader.onload = (event: any) => {\n        this.loadImage(event.target.result, file.type, cropperSettings)\n          .then(resolve)\n          .catch(reject);\n      };\n      fileReader.readAsDataURL(file);\n    });\n  }\n\n  private loadImage(imageBase64: string, imageType: string, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    if (!this.isValidImageType(imageType)) {\n      return Promise.reject(new Error('Invalid image type'));\n    }\n    return this.loadBase64Image(imageBase64, cropperSettings);\n  }\n\n  private isValidImageType(type: string): boolean {\n    return /image\\/(png|jpg|jpeg|bmp|gif|tiff|webp)/.test(type);\n  }\n\n  loadImageFromURL(url: string, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    return new Promise((resolve, reject) => {\n      const img = new Image();\n      img.onerror = () => reject;\n      img.onload = () => {\n        const canvas = document.createElement('canvas');\n        const context = canvas.getContext('2d');\n        canvas.width = img.width;\n        canvas.height = img.height;\n        context.drawImage(img, 0, 0);\n        this.loadBase64Image(canvas.toDataURL(), cropperSettings).then(resolve);\n      };\n      img.crossOrigin = 'anonymous';\n      img.src = url;\n    });\n  }\n\n  loadBase64Image(imageBase64: string, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    return new Promise<LoadImageBase64>((resolve, reject) => {\n      const originalImage = new Image();\n      originalImage.onload = () => resolve({\n        originalImage,\n        originalBase64: imageBase64\n      });\n      originalImage.onerror = reject;\n      originalImage.src = imageBase64;\n    }).then((res: LoadImageBase64) => this.transformImageBase64(res, cropperSettings));\n  }\n\n  private async transformImageBase64(res: LoadImageBase64, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    const autoRotate = await this.autoRotateSupported;\n    const exifTransform = await getTransformationsFromExifData(autoRotate ? -1 : res.originalBase64);\n    if (!res.originalImage || !res.originalImage.complete) {\n      return Promise.reject(new Error('No image loaded'));\n    }\n    const loadedImage = {\n      original: {\n        base64: res.originalBase64,\n        image: res.originalImage,\n        size: {\n          width: res.originalImage.naturalWidth,\n          height: res.originalImage.naturalHeight\n        }\n      },\n      exifTransform\n    };\n    return this.transformLoadedImage(loadedImage, cropperSettings);\n  }\n\n  async transformLoadedImage(loadedImage: Partial<LoadedImage>, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    const canvasRotation = cropperSettings.canvasRotation + loadedImage.exifTransform.rotate;\n    const originalSize = {\n      width: loadedImage.original.image.naturalWidth,\n      height: loadedImage.original.image.naturalHeight\n    };\n    if (canvasRotation === 0 && !loadedImage.exifTransform.flip && !cropperSettings.containWithinAspectRatio) {\n      return {\n        original: {\n          base64: loadedImage.original.base64,\n          image: loadedImage.original.image,\n          size: {...originalSize}\n        },\n        transformed: {\n          base64: loadedImage.original.base64,\n          image: loadedImage.original.image,\n          size: {...originalSize}\n        },\n        exifTransform: loadedImage.exifTransform\n      };\n    }\n\n    const transformedSize = this.getTransformedSize(originalSize, loadedImage.exifTransform, cropperSettings);\n    const canvas = document.createElement('canvas');\n    canvas.width = transformedSize.width;\n    canvas.height = transformedSize.height;\n    const ctx = canvas.getContext('2d');\n    ctx.setTransform(\n      loadedImage.exifTransform.flip ? -1 : 1,\n      0,\n      0,\n      1,\n      canvas.width / 2,\n      canvas.height / 2\n    );\n    ctx.rotate(Math.PI * (canvasRotation / 2));\n    ctx.drawImage(\n      loadedImage.original.image,\n      -originalSize.width / 2,\n      -originalSize.height / 2\n    );\n    const transformedBase64 = canvas.toDataURL();\n    const transformedImage = await this.loadImageFromBase64(transformedBase64);\n    return {\n      original: {\n        base64: loadedImage.original.base64,\n        image: loadedImage.original.image,\n        size: {...originalSize}\n      },\n      transformed: {\n        base64: transformedBase64,\n        image: transformedImage,\n        size: {\n          width: transformedImage.width,\n          height: transformedImage.height\n        }\n      },\n      exifTransform: loadedImage.exifTransform\n    };\n  }\n\n  private loadImageFromBase64(imageBase64: string): Promise<HTMLImageElement> {\n    return new Promise<HTMLImageElement>(((resolve, reject) => {\n      const image = new Image();\n      image.onload = () => resolve(image);\n      image.onerror = reject;\n      image.src = imageBase64;\n    }));\n  }\n\n  private getTransformedSize(\n    originalSize: { width: number, height: number },\n    exifTransform: ExifTransform,\n    cropperSettings: CropperSettings\n  ): Dimensions {\n    const canvasRotation = cropperSettings.canvasRotation + exifTransform.rotate;\n    if (cropperSettings.containWithinAspectRatio) {\n      if (canvasRotation % 2) {\n        const minWidthToContain = originalSize.width * cropperSettings.aspectRatio;\n        const minHeightToContain = originalSize.height / cropperSettings.aspectRatio;\n        return {\n          width: Math.max(originalSize.height, minWidthToContain),\n          height: Math.max(originalSize.width, minHeightToContain)\n        };\n      } else {\n        const minWidthToContain = originalSize.height * cropperSettings.aspectRatio;\n        const minHeightToContain = originalSize.width / cropperSettings.aspectRatio;\n        return {\n          width: Math.max(originalSize.width, minWidthToContain),\n          height: Math.max(originalSize.height, minHeightToContain)\n        };\n      }\n    }\n\n    if (canvasRotation % 2) {\n      return {\n        height: originalSize.width,\n        width: originalSize.height\n      };\n    }\n    return {\n      width: originalSize.width,\n      height: originalSize.height\n    };\n  }\n}\n"]}
@@ -1,26 +0,0 @@
1
- /**
2
- * @fileoverview added by tsickle
3
- * Generated from: lib/utils/blob.utils.ts
4
- * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
- */
6
- /**
7
- * @param {?} base64Image
8
- * @return {?}
9
- */
10
- export function base64ToFile(base64Image) {
11
- /** @type {?} */
12
- const split = base64Image.split(',');
13
- /** @type {?} */
14
- const type = split[0].replace('data:', '').replace(';base64', '');
15
- /** @type {?} */
16
- const byteString = atob(split[1]);
17
- /** @type {?} */
18
- const ab = new ArrayBuffer(byteString.length);
19
- /** @type {?} */
20
- const ia = new Uint8Array(ab);
21
- for (let i = 0; i < byteString.length; i += 1) {
22
- ia[i] = byteString.charCodeAt(i);
23
- }
24
- return new Blob([ab], { type });
25
- }
26
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmxvYi51dGlscy5qcyIsInNvdXJjZVJvb3QiOiIvVXNlcnMvbWF3aS9Qcm9qZWN0cy9uZ3gtaW1hZ2UtY3JvcHBlci9wcm9qZWN0cy9uZ3gtaW1hZ2UtY3JvcHBlci9zcmMvIiwic291cmNlcyI6WyJsaWIvdXRpbHMvYmxvYi51dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFBQSxNQUFNLFVBQVUsWUFBWSxDQUFDLFdBQW1COztVQUN4QyxLQUFLLEdBQUcsV0FBVyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUM7O1VBQzlCLElBQUksR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQyxPQUFPLENBQUMsU0FBUyxFQUFFLEVBQUUsQ0FBQzs7VUFDM0QsVUFBVSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7O1VBQzNCLEVBQUUsR0FBRyxJQUFJLFdBQVcsQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDOztVQUN2QyxFQUFFLEdBQUcsSUFBSSxVQUFVLENBQUMsRUFBRSxDQUFDO0lBQzdCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxVQUFVLENBQUMsTUFBTSxFQUFFLENBQUMsSUFBSSxDQUFDLEVBQUU7UUFDN0MsRUFBRSxDQUFDLENBQUMsQ0FBQyxHQUFHLFVBQVUsQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUM7S0FDbEM7SUFDRCxPQUFPLElBQUksSUFBSSxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBQyxJQUFJLEVBQUMsQ0FBQyxDQUFDO0FBQ2hDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgZnVuY3Rpb24gYmFzZTY0VG9GaWxlKGJhc2U2NEltYWdlOiBzdHJpbmcpOiBCbG9iIHtcbiAgY29uc3Qgc3BsaXQgPSBiYXNlNjRJbWFnZS5zcGxpdCgnLCcpO1xuICBjb25zdCB0eXBlID0gc3BsaXRbMF0ucmVwbGFjZSgnZGF0YTonLCAnJykucmVwbGFjZSgnO2Jhc2U2NCcsICcnKTtcbiAgY29uc3QgYnl0ZVN0cmluZyA9IGF0b2Ioc3BsaXRbMV0pO1xuICBjb25zdCBhYiA9IG5ldyBBcnJheUJ1ZmZlcihieXRlU3RyaW5nLmxlbmd0aCk7XG4gIGNvbnN0IGlhID0gbmV3IFVpbnQ4QXJyYXkoYWIpO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IGJ5dGVTdHJpbmcubGVuZ3RoOyBpICs9IDEpIHtcbiAgICBpYVtpXSA9IGJ5dGVTdHJpbmcuY2hhckNvZGVBdChpKTtcbiAgfVxuICByZXR1cm4gbmV3IEJsb2IoW2FiXSwge3R5cGV9KTtcbn1cbiJdfQ==
@@ -1,128 +0,0 @@
1
- /**
2
- * @fileoverview added by tsickle
3
- * Generated from: lib/utils/exif.utils.ts
4
- * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
- */
6
- // Black 2x1 JPEG, with the following meta information set:
7
- // - EXIF Orientation: 6 (Rotated 90° CCW)
8
- // Source: https://github.com/blueimp/JavaScript-Load-Image
9
- /** @type {?} */
10
- const testAutoOrientationImageURL = 'data:image/jpeg;base64,/9j/4QAiRXhpZgAATU0AKgAAAAgAAQESAAMAAAABAAYAAAA' +
11
- 'AAAD/2wCEAAEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBA' +
12
- 'QEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQE' +
13
- 'BAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAf/AABEIAAEAAgMBEQACEQEDEQH/x' +
14
- 'ABKAAEAAAAAAAAAAAAAAAAAAAALEAEAAAAAAAAAAAAAAAAAAAAAAQEAAAAAAAAAAAAAAAA' +
15
- 'AAAAAEQEAAAAAAAAAAAAAAAAAAAAA/9oADAMBAAIRAxEAPwA/8H//2Q==';
16
- /**
17
- * @return {?}
18
- */
19
- export function supportsAutomaticRotation() {
20
- return new Promise((/**
21
- * @param {?} resolve
22
- * @return {?}
23
- */
24
- (resolve) => {
25
- /** @type {?} */
26
- const img = new Image();
27
- img.onload = (/**
28
- * @return {?}
29
- */
30
- () => {
31
- // Check if browser supports automatic image orientation:
32
- /** @type {?} */
33
- const supported = img.width === 1 && img.height === 2;
34
- resolve(supported);
35
- });
36
- img.src = testAutoOrientationImageURL;
37
- }));
38
- }
39
- /**
40
- * @param {?} exifRotationOrBase64Image
41
- * @return {?}
42
- */
43
- export function getTransformationsFromExifData(exifRotationOrBase64Image) {
44
- if (typeof exifRotationOrBase64Image === 'string') {
45
- exifRotationOrBase64Image = getExifRotation(exifRotationOrBase64Image);
46
- }
47
- switch (exifRotationOrBase64Image) {
48
- case 2:
49
- return { rotate: 0, flip: true };
50
- case 3:
51
- return { rotate: 2, flip: false };
52
- case 4:
53
- return { rotate: 2, flip: true };
54
- case 5:
55
- return { rotate: 1, flip: true };
56
- case 6:
57
- return { rotate: 1, flip: false };
58
- case 7:
59
- return { rotate: 3, flip: true };
60
- case 8:
61
- return { rotate: 3, flip: false };
62
- default:
63
- return { rotate: 0, flip: false };
64
- }
65
- }
66
- /**
67
- * @param {?} imageBase64
68
- * @return {?}
69
- */
70
- function getExifRotation(imageBase64) {
71
- /** @type {?} */
72
- const view = new DataView(base64ToArrayBuffer(imageBase64));
73
- if (view.getUint16(0, false) !== 0xFFD8) {
74
- return -2;
75
- }
76
- /** @type {?} */
77
- const length = view.byteLength;
78
- /** @type {?} */
79
- let offset = 2;
80
- while (offset < length) {
81
- if (view.getUint16(offset + 2, false) <= 8)
82
- return -1;
83
- /** @type {?} */
84
- const marker = view.getUint16(offset, false);
85
- offset += 2;
86
- if (marker == 0xFFE1) {
87
- if (view.getUint32(offset += 2, false) !== 0x45786966) {
88
- return -1;
89
- }
90
- /** @type {?} */
91
- const little = view.getUint16(offset += 6, false) == 0x4949;
92
- offset += view.getUint32(offset + 4, little);
93
- /** @type {?} */
94
- const tags = view.getUint16(offset, little);
95
- offset += 2;
96
- for (let i = 0; i < tags; i++) {
97
- if (view.getUint16(offset + (i * 12), little) == 0x0112) {
98
- return view.getUint16(offset + (i * 12) + 8, little);
99
- }
100
- }
101
- }
102
- else if ((marker & 0xFF00) !== 0xFF00) {
103
- break;
104
- }
105
- else {
106
- offset += view.getUint16(offset, false);
107
- }
108
- }
109
- return -1;
110
- }
111
- /**
112
- * @param {?} imageBase64
113
- * @return {?}
114
- */
115
- function base64ToArrayBuffer(imageBase64) {
116
- imageBase64 = imageBase64.replace(/^data\:([^\;]+)\;base64,/gmi, '');
117
- /** @type {?} */
118
- const binaryString = atob(imageBase64);
119
- /** @type {?} */
120
- const len = binaryString.length;
121
- /** @type {?} */
122
- const bytes = new Uint8Array(len);
123
- for (let i = 0; i < len; i++) {
124
- bytes[i] = binaryString.charCodeAt(i);
125
- }
126
- return bytes.buffer;
127
- }
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exif.utils.js","sourceRoot":"/Users/mawi/Projects/ngx-image-cropper/projects/ngx-image-cropper/src/","sources":["lib/utils/exif.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { ExifTransform } from '../interfaces/exif-transform.interface';\n\n// Black 2x1 JPEG, with the following meta information set:\n// - EXIF Orientation: 6 (Rotated 90° CCW)\n// Source: https://github.com/blueimp/JavaScript-Load-Image\nconst testAutoOrientationImageURL =\n  'data:image/jpeg;base64,/9j/4QAiRXhpZgAATU0AKgAAAAgAAQESAAMAAAABAAYAAAA' +\n  'AAAD/2wCEAAEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBA' +\n  'QEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQE' +\n  'BAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAf/AABEIAAEAAgMBEQACEQEDEQH/x' +\n  'ABKAAEAAAAAAAAAAAAAAAAAAAALEAEAAAAAAAAAAAAAAAAAAAAAAQEAAAAAAAAAAAAAAAA' +\n  'AAAAAEQEAAAAAAAAAAAAAAAAAAAAA/9oADAMBAAIRAxEAPwA/8H//2Q==';\n\nexport function supportsAutomaticRotation(): Promise<boolean> {\n  return new Promise((resolve) => {\n    const img = new Image();\n    img.onload = () => {\n      // Check if browser supports automatic image orientation:\n      const supported = img.width === 1 && img.height === 2;\n      resolve(supported);\n    };\n    img.src = testAutoOrientationImageURL;\n  });\n}\n\nexport function getTransformationsFromExifData(exifRotationOrBase64Image: number | string): ExifTransform {\n  if (typeof exifRotationOrBase64Image === 'string') {\n    exifRotationOrBase64Image = getExifRotation(exifRotationOrBase64Image);\n  }\n  switch (exifRotationOrBase64Image) {\n    case 2:\n      return { rotate: 0, flip: true };\n    case 3:\n      return { rotate: 2, flip: false };\n    case 4:\n      return { rotate: 2, flip: true };\n    case 5:\n      return { rotate: 1, flip: true };\n    case 6:\n      return { rotate: 1, flip: false };\n    case 7:\n      return { rotate: 3, flip: true };\n    case 8:\n      return { rotate: 3, flip: false };\n    default:\n      return { rotate: 0, flip: false };\n  }\n}\n\nfunction getExifRotation(imageBase64: string): number {\n  const view = new DataView(base64ToArrayBuffer(imageBase64));\n  if (view.getUint16(0, false) !== 0xFFD8) {\n    return -2;\n  }\n  const length = view.byteLength;\n  let offset = 2;\n  while (offset < length) {\n    if (view.getUint16(offset + 2, false) <= 8) return -1;\n    const marker = view.getUint16(offset, false);\n    offset += 2;\n    if (marker == 0xFFE1) {\n      if (view.getUint32(offset += 2, false) !== 0x45786966) {\n        return -1;\n      }\n\n      const little = view.getUint16(offset += 6, false) == 0x4949;\n      offset += view.getUint32(offset + 4, little);\n      const tags = view.getUint16(offset, little);\n      offset += 2;\n      for (let i = 0; i < tags; i++) {\n        if (view.getUint16(offset + (i * 12), little) == 0x0112) {\n          return view.getUint16(offset + (i * 12) + 8, little);\n        }\n      }\n    } else if ((marker & 0xFF00) !== 0xFF00) {\n      break;\n    } else {\n      offset += view.getUint16(offset, false);\n    }\n  }\n  return -1;\n}\n\nfunction base64ToArrayBuffer(imageBase64: string) {\n  imageBase64 = imageBase64.replace(/^data\\:([^\\;]+)\\;base64,/gmi, '');\n  const binaryString = atob(imageBase64);\n  const len = binaryString.length;\n  const bytes = new Uint8Array(len);\n  for (let i = 0; i < len; i++) {\n    bytes[i] = binaryString.charCodeAt(i);\n  }\n  return bytes.buffer;\n}\n"]}
@@ -1,29 +0,0 @@
1
- /**
2
- * @fileoverview added by tsickle
3
- * Generated from: lib/utils/hammer.utils.ts
4
- * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
- */
6
- /**
7
- * \@docs-private
8
- * @record
9
- */
10
- export function HammerManager() { }
11
- if (false) {
12
- /**
13
- * @param {?} eventName
14
- * @return {?}
15
- */
16
- HammerManager.prototype.get = function (eventName) { };
17
- /**
18
- * @param {?} options
19
- * @return {?}
20
- */
21
- HammerManager.prototype.set = function (options) { };
22
- /**
23
- * @param {?} eventName
24
- * @param {?} handler
25
- * @return {?}
26
- */
27
- HammerManager.prototype.on = function (eventName, handler) { };
28
- }
29
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFtbWVyLnV0aWxzLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9tYXdpL1Byb2plY3RzL25neC1pbWFnZS1jcm9wcGVyL3Byb2plY3RzL25neC1pbWFnZS1jcm9wcGVyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi91dGlscy9oYW1tZXIudXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7O0FBR0EsbUNBTUM7Ozs7OztJQUxDLHVEQUFzQzs7Ozs7SUFFdEMscURBQWlDOzs7Ozs7SUFFakMsK0RBQWlEIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgSGFtbWVyU3RhdGljID0gbmV3KGVsZW1lbnQ6IEhUTUxFbGVtZW50IHwgU1ZHRWxlbWVudCwgb3B0aW9ucz86IGFueSkgPT4gSGFtbWVyTWFuYWdlcjtcblxuLyoqIEBkb2NzLXByaXZhdGUgKi9cbmV4cG9ydCBpbnRlcmZhY2UgSGFtbWVyTWFuYWdlciB7XG4gIGdldChldmVudE5hbWU6IHN0cmluZyk6IEhhbW1lck1hbmFnZXI7XG5cbiAgc2V0KG9wdGlvbnM6IGFueSk6IEhhbW1lck1hbmFnZXI7XG5cbiAgb24oZXZlbnROYW1lOiBzdHJpbmcsIGhhbmRsZXI6IChldjogYW55KSA9PiBhbnkpO1xufVxuIl19
@@ -1,58 +0,0 @@
1
- /**
2
- * @fileoverview added by tsickle
3
- * Generated from: lib/utils/keyboard.utils.ts
4
- * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
- */
6
- /**
7
- * @param {?} key
8
- * @return {?}
9
- */
10
- export function getPositionForKey(key) {
11
- switch (key) {
12
- case 'ArrowUp':
13
- return 'top';
14
- case 'ArrowRight':
15
- return 'right';
16
- case 'ArrowDown':
17
- return 'bottom';
18
- case 'ArrowLeft':
19
- default:
20
- return 'left';
21
- }
22
- }
23
- /**
24
- * @param {?} key
25
- * @return {?}
26
- */
27
- export function getInvertedPositionForKey(key) {
28
- switch (key) {
29
- case 'ArrowUp':
30
- return 'bottom';
31
- case 'ArrowRight':
32
- return 'left';
33
- case 'ArrowDown':
34
- return 'top';
35
- case 'ArrowLeft':
36
- default:
37
- return 'right';
38
- }
39
- }
40
- /**
41
- * @param {?} key
42
- * @param {?} stepSize
43
- * @return {?}
44
- */
45
- export function getEventForKey(key, stepSize) {
46
- switch (key) {
47
- case 'ArrowUp':
48
- return { clientX: 0, clientY: stepSize * -1 };
49
- case 'ArrowRight':
50
- return { clientX: stepSize, clientY: 0 };
51
- case 'ArrowDown':
52
- return { clientX: 0, clientY: stepSize };
53
- case 'ArrowLeft':
54
- default:
55
- return { clientX: stepSize * -1, clientY: 0 };
56
- }
57
- }
58
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoia2V5Ym9hcmQudXRpbHMuanMiLCJzb3VyY2VSb290IjoiL1VzZXJzL21hd2kvUHJvamVjdHMvbmd4LWltYWdlLWNyb3BwZXIvcHJvamVjdHMvbmd4LWltYWdlLWNyb3BwZXIvc3JjLyIsInNvdXJjZXMiOlsibGliL3V0aWxzL2tleWJvYXJkLnV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUFBLE1BQU0sVUFBVSxpQkFBaUIsQ0FBQyxHQUFXO0lBQzNDLFFBQVEsR0FBRyxFQUFFO1FBQ1gsS0FBSyxTQUFTO1lBQ1osT0FBTyxLQUFLLENBQUM7UUFDZixLQUFLLFlBQVk7WUFDZixPQUFPLE9BQU8sQ0FBQztRQUNqQixLQUFLLFdBQVc7WUFDZCxPQUFPLFFBQVEsQ0FBQztRQUNsQixLQUFLLFdBQVcsQ0FBQztRQUNqQjtZQUNFLE9BQU8sTUFBTSxDQUFDO0tBQ2pCO0FBQ0gsQ0FBQzs7Ozs7QUFFRCxNQUFNLFVBQVUseUJBQXlCLENBQUMsR0FBVztJQUNuRCxRQUFRLEdBQUcsRUFBRTtRQUNYLEtBQUssU0FBUztZQUNaLE9BQU8sUUFBUSxDQUFDO1FBQ2xCLEtBQUssWUFBWTtZQUNmLE9BQU8sTUFBTSxDQUFDO1FBQ2hCLEtBQUssV0FBVztZQUNkLE9BQU8sS0FBSyxDQUFDO1FBQ2YsS0FBSyxXQUFXLENBQUM7UUFDakI7WUFDRSxPQUFPLE9BQU8sQ0FBQztLQUNsQjtBQUNILENBQUM7Ozs7OztBQUVELE1BQU0sVUFBVSxjQUFjLENBQUMsR0FBVyxFQUFFLFFBQWdCO0lBQzFELFFBQVEsR0FBRyxFQUFFO1FBQ1gsS0FBSyxTQUFTO1lBQ1osT0FBTyxFQUFDLE9BQU8sRUFBRSxDQUFDLEVBQUUsT0FBTyxFQUFFLFFBQVEsR0FBRyxDQUFDLENBQUMsRUFBQyxDQUFDO1FBQzlDLEtBQUssWUFBWTtZQUNmLE9BQU8sRUFBQyxPQUFPLEVBQUUsUUFBUSxFQUFFLE9BQU8sRUFBRSxDQUFDLEVBQUMsQ0FBQztRQUN6QyxLQUFLLFdBQVc7WUFDZCxPQUFPLEVBQUMsT0FBTyxFQUFFLENBQUMsRUFBRSxPQUFPLEVBQUUsUUFBUSxFQUFDLENBQUM7UUFDekMsS0FBSyxXQUFXLENBQUM7UUFDakI7WUFDRSxPQUFPLEVBQUMsT0FBTyxFQUFFLFFBQVEsR0FBRyxDQUFDLENBQUMsRUFBRSxPQUFPLEVBQUUsQ0FBQyxFQUFDLENBQUM7S0FDL0M7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIGdldFBvc2l0aW9uRm9yS2V5KGtleTogc3RyaW5nKTogc3RyaW5nIHtcbiAgc3dpdGNoIChrZXkpIHtcbiAgICBjYXNlICdBcnJvd1VwJzpcbiAgICAgIHJldHVybiAndG9wJztcbiAgICBjYXNlICdBcnJvd1JpZ2h0JzpcbiAgICAgIHJldHVybiAncmlnaHQnO1xuICAgIGNhc2UgJ0Fycm93RG93bic6XG4gICAgICByZXR1cm4gJ2JvdHRvbSc7XG4gICAgY2FzZSAnQXJyb3dMZWZ0JzpcbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuICdsZWZ0JztcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0SW52ZXJ0ZWRQb3NpdGlvbkZvcktleShrZXk6IHN0cmluZyk6IHN0cmluZyB7XG4gIHN3aXRjaCAoa2V5KSB7XG4gICAgY2FzZSAnQXJyb3dVcCc6XG4gICAgICByZXR1cm4gJ2JvdHRvbSc7XG4gICAgY2FzZSAnQXJyb3dSaWdodCc6XG4gICAgICByZXR1cm4gJ2xlZnQnO1xuICAgIGNhc2UgJ0Fycm93RG93bic6XG4gICAgICByZXR1cm4gJ3RvcCc7XG4gICAgY2FzZSAnQXJyb3dMZWZ0JzpcbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuICdyaWdodCc7XG4gIH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldEV2ZW50Rm9yS2V5KGtleTogc3RyaW5nLCBzdGVwU2l6ZTogbnVtYmVyKTogYW55IHtcbiAgc3dpdGNoIChrZXkpIHtcbiAgICBjYXNlICdBcnJvd1VwJzpcbiAgICAgIHJldHVybiB7Y2xpZW50WDogMCwgY2xpZW50WTogc3RlcFNpemUgKiAtMX07XG4gICAgY2FzZSAnQXJyb3dSaWdodCc6XG4gICAgICByZXR1cm4ge2NsaWVudFg6IHN0ZXBTaXplLCBjbGllbnRZOiAwfTtcbiAgICBjYXNlICdBcnJvd0Rvd24nOlxuICAgICAgcmV0dXJuIHtjbGllbnRYOiAwLCBjbGllbnRZOiBzdGVwU2l6ZX07XG4gICAgY2FzZSAnQXJyb3dMZWZ0JzpcbiAgICBkZWZhdWx0OlxuICAgICAgcmV0dXJuIHtjbGllbnRYOiBzdGVwU2l6ZSAqIC0xLCBjbGllbnRZOiAwfTtcbiAgfVxufVxuIl19