ngx-image-cropper 7.2.1 → 8.1.0

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 (59) hide show
  1. package/README.md +56 -72
  2. package/esm2022/lib/component/cropper.state.mjs +188 -0
  3. package/esm2022/lib/component/image-cropper.component.mjs +523 -0
  4. package/esm2022/lib/interfaces/basic-event.interface.mjs +2 -0
  5. package/esm2022/lib/interfaces/cropper-options.interface.mjs +2 -0
  6. package/esm2022/lib/interfaces/index.mjs +2 -0
  7. package/esm2022/lib/interfaces/move-start.interface.mjs +8 -0
  8. package/esm2022/lib/services/crop.service.mjs +139 -0
  9. package/esm2022/lib/services/load-image.service.mjs +196 -0
  10. package/esm2022/lib/utils/cropper-position.utils.mjs +239 -0
  11. package/{esm2020 → esm2022}/lib/utils/keyboard.utils.mjs +1 -1
  12. package/esm2022/public-api.mjs +7 -0
  13. package/fesm2022/ngx-image-cropper.mjs +1500 -0
  14. package/fesm2022/ngx-image-cropper.mjs.map +1 -0
  15. package/index.d.ts +5 -7
  16. package/lib/component/cropper.state.d.ts +29 -0
  17. package/lib/component/image-cropper.component.d.ts +58 -65
  18. package/lib/interfaces/basic-event.interface.d.ts +4 -0
  19. package/lib/interfaces/cropper-options.interface.d.ts +7 -7
  20. package/lib/interfaces/index.d.ts +0 -1
  21. package/lib/interfaces/move-start.interface.d.ts +4 -6
  22. package/lib/services/crop.service.d.ts +6 -9
  23. package/lib/services/load-image.service.d.ts +5 -8
  24. package/lib/utils/cropper-position.utils.d.ts +11 -0
  25. package/lib/utils/keyboard.utils.d.ts +5 -3
  26. package/package.json +10 -18
  27. package/public-api.d.ts +6 -1
  28. package/esm2020/index.mjs +0 -8
  29. package/esm2020/lib/component/image-cropper.component.mjs +0 -635
  30. package/esm2020/lib/image-cropper.module.mjs +0 -26
  31. package/esm2020/lib/interfaces/cropper-options.interface.mjs +0 -2
  32. package/esm2020/lib/interfaces/cropper.settings.mjs +0 -54
  33. package/esm2020/lib/interfaces/index.mjs +0 -2
  34. package/esm2020/lib/interfaces/move-start.interface.mjs +0 -8
  35. package/esm2020/lib/services/crop.service.mjs +0 -147
  36. package/esm2020/lib/services/cropper-position.service.mjs +0 -200
  37. package/esm2020/lib/services/load-image.service.mjs +0 -201
  38. package/esm2020/lib/utils/hammer.utils.mjs +0 -2
  39. package/esm2020/public-api.mjs +0 -2
  40. package/fesm2015/ngx-image-cropper.mjs +0 -1484
  41. package/fesm2015/ngx-image-cropper.mjs.map +0 -1
  42. package/fesm2020/ngx-image-cropper.mjs +0 -1470
  43. package/fesm2020/ngx-image-cropper.mjs.map +0 -1
  44. package/lib/image-cropper.module.d.ts +0 -8
  45. package/lib/interfaces/cropper.settings.d.ts +0 -38
  46. package/lib/services/cropper-position.service.d.ts +0 -14
  47. package/lib/utils/hammer.utils.d.ts +0 -7
  48. package/ngx-image-cropper.d.ts +0 -5
  49. /package/{esm2020 → esm2022}/lib/interfaces/cropper-position.interface.mjs +0 -0
  50. /package/{esm2020 → esm2022}/lib/interfaces/dimensions.interface.mjs +0 -0
  51. /package/{esm2020 → esm2022}/lib/interfaces/exif-transform.interface.mjs +0 -0
  52. /package/{esm2020 → esm2022}/lib/interfaces/image-cropped-event.interface.mjs +0 -0
  53. /package/{esm2020 → esm2022}/lib/interfaces/image-transform.interface.mjs +0 -0
  54. /package/{esm2020 → esm2022}/lib/interfaces/loaded-image.interface.mjs +0 -0
  55. /package/{esm2020 → esm2022}/lib/utils/blob.utils.mjs +0 -0
  56. /package/{esm2020 → esm2022}/lib/utils/exif.utils.mjs +0 -0
  57. /package/{esm2020 → esm2022}/lib/utils/percentage.utils.mjs +0 -0
  58. /package/{esm2020 → esm2022}/lib/utils/resize.utils.mjs +0 -0
  59. /package/{esm2020 → esm2022}/ngx-image-cropper.mjs +0 -0
@@ -1,201 +0,0 @@
1
- import { Injectable } from '@angular/core';
2
- import { getTransformationsFromExifData, supportsAutomaticRotation } from '../utils/exif.utils';
3
- import * as i0 from "@angular/core";
4
- export class LoadImageService {
5
- constructor() {
6
- this.autoRotateSupported = supportsAutomaticRotation();
7
- }
8
- async loadImageFile(file, cropperSettings) {
9
- const arrayBuffer = await file.arrayBuffer();
10
- return await this.checkImageTypeAndLoadImageFromArrayBuffer(arrayBuffer, file.type, cropperSettings);
11
- }
12
- checkImageTypeAndLoadImageFromArrayBuffer(arrayBuffer, imageType, cropperSettings) {
13
- if (!this.isValidImageType(imageType)) {
14
- return Promise.reject(new Error('Invalid image type'));
15
- }
16
- return this.loadImageFromArrayBuffer(arrayBuffer, cropperSettings, imageType);
17
- }
18
- isValidImageType(type) {
19
- return /image\/(png|jpg|jpeg|bmp|gif|tiff|svg|webp|x-icon|vnd.microsoft.icon)/.test(type);
20
- }
21
- async loadImageFromURL(url, cropperSettings) {
22
- const res = await fetch(url);
23
- const blob = await res.blob();
24
- const buffer = await blob.arrayBuffer();
25
- return await this.loadImageFromArrayBuffer(buffer, cropperSettings, blob.type);
26
- }
27
- loadBase64Image(imageBase64, cropperSettings) {
28
- const arrayBuffer = this.base64ToArrayBuffer(imageBase64);
29
- return this.loadImageFromArrayBuffer(arrayBuffer, cropperSettings);
30
- }
31
- base64ToArrayBuffer(imageBase64) {
32
- imageBase64 = imageBase64.replace(/^data:([^;]+);base64,/gmi, '');
33
- const binaryString = atob(imageBase64);
34
- const len = binaryString.length;
35
- const bytes = new Uint8Array(len);
36
- for (let i = 0; i < len; i++) {
37
- bytes[i] = binaryString.charCodeAt(i);
38
- }
39
- return bytes.buffer;
40
- }
41
- async loadImageFromArrayBuffer(arrayBuffer, cropperSettings, imageType) {
42
- const res = await new Promise(async (resolve, reject) => {
43
- try {
44
- const blob = new Blob([arrayBuffer], imageType ? { type: imageType } : undefined);
45
- const objectUrl = URL.createObjectURL(blob);
46
- const originalImage = new Image();
47
- const isSvg = imageType === 'image/svg+xml';
48
- const originalImageSize = isSvg ? await this.getSvgImageSize(blob) : undefined;
49
- originalImage.onload = () => resolve({
50
- originalImage,
51
- originalImageSize,
52
- originalObjectUrl: objectUrl,
53
- originalArrayBuffer: arrayBuffer
54
- });
55
- originalImage.onerror = reject;
56
- originalImage.src = objectUrl;
57
- }
58
- catch (e) {
59
- reject(e);
60
- }
61
- });
62
- return await this.transformImageFromArrayBuffer(res, cropperSettings, res.originalImageSize != null);
63
- }
64
- async getSvgImageSize(blob) {
65
- const parser = new DOMParser();
66
- const doc = parser.parseFromString(await blob.text(), 'image/svg+xml');
67
- const svgElement = doc.querySelector('svg');
68
- if (!svgElement) {
69
- throw Error('Failed to parse SVG image');
70
- }
71
- const widthAttr = svgElement.getAttribute('width');
72
- const heightAttr = svgElement.getAttribute('height');
73
- if (widthAttr && heightAttr) {
74
- return null;
75
- }
76
- const viewBoxAttr = svgElement.getAttribute('viewBox')
77
- || svgElement.getAttribute('viewbox');
78
- if (viewBoxAttr) {
79
- const viewBox = viewBoxAttr.split(' ');
80
- return {
81
- width: +viewBox[2],
82
- height: +viewBox[3]
83
- };
84
- }
85
- throw Error('Failed to load SVG image. SVG must have width + height or viewBox definition.');
86
- }
87
- async transformImageFromArrayBuffer(res, cropperSettings, forceTransform = false) {
88
- const autoRotate = await this.autoRotateSupported;
89
- const exifTransform = getTransformationsFromExifData(autoRotate ? -1 : res.originalArrayBuffer);
90
- if (!res.originalImage || !res.originalImage.complete) {
91
- return Promise.reject(new Error('No image loaded'));
92
- }
93
- const loadedImage = {
94
- original: {
95
- objectUrl: res.originalObjectUrl,
96
- image: res.originalImage,
97
- size: res.originalImageSize ?? {
98
- width: res.originalImage.naturalWidth,
99
- height: res.originalImage.naturalHeight
100
- }
101
- },
102
- exifTransform
103
- };
104
- return this.transformLoadedImage(loadedImage, cropperSettings, forceTransform);
105
- }
106
- async transformLoadedImage(loadedImage, cropperSettings, forceTransform = false) {
107
- const canvasRotation = cropperSettings.canvasRotation + loadedImage.exifTransform.rotate;
108
- const originalSize = loadedImage.original.size;
109
- if (!forceTransform && canvasRotation === 0 && !loadedImage.exifTransform.flip && !cropperSettings.containWithinAspectRatio) {
110
- return {
111
- original: {
112
- objectUrl: loadedImage.original.objectUrl,
113
- image: loadedImage.original.image,
114
- size: { ...originalSize }
115
- },
116
- transformed: {
117
- objectUrl: loadedImage.original.objectUrl,
118
- image: loadedImage.original.image,
119
- size: { ...originalSize }
120
- },
121
- exifTransform: loadedImage.exifTransform
122
- };
123
- }
124
- const transformedSize = this.getTransformedSize(originalSize, loadedImage.exifTransform, cropperSettings);
125
- const canvas = document.createElement('canvas');
126
- canvas.width = transformedSize.width;
127
- canvas.height = transformedSize.height;
128
- const ctx = canvas.getContext('2d');
129
- ctx?.setTransform(loadedImage.exifTransform.flip ? -1 : 1, 0, 0, 1, canvas.width / 2, canvas.height / 2);
130
- ctx?.rotate(Math.PI * (canvasRotation / 2));
131
- ctx?.drawImage(loadedImage.original.image, -originalSize.width / 2, -originalSize.height / 2);
132
- const blob = await new Promise(resolve => canvas.toBlob(resolve, cropperSettings.format));
133
- if (!blob) {
134
- throw new Error('Failed to get Blob for transformed image.');
135
- }
136
- const objectUrl = URL.createObjectURL(blob);
137
- const transformedImage = await this.loadImageFromObjectUrl(objectUrl);
138
- return {
139
- original: {
140
- objectUrl: loadedImage.original.objectUrl,
141
- image: loadedImage.original.image,
142
- size: { ...originalSize }
143
- },
144
- transformed: {
145
- objectUrl: objectUrl,
146
- image: transformedImage,
147
- size: {
148
- width: transformedImage.width,
149
- height: transformedImage.height
150
- }
151
- },
152
- exifTransform: loadedImage.exifTransform
153
- };
154
- }
155
- loadImageFromObjectUrl(objectUrl) {
156
- return new Promise(((resolve, reject) => {
157
- const image = new Image();
158
- image.onload = () => resolve(image);
159
- image.onerror = reject;
160
- image.src = objectUrl;
161
- }));
162
- }
163
- getTransformedSize(originalSize, exifTransform, cropperSettings) {
164
- const canvasRotation = cropperSettings.canvasRotation + exifTransform.rotate;
165
- if (cropperSettings.containWithinAspectRatio) {
166
- if (canvasRotation % 2) {
167
- const minWidthToContain = originalSize.width * cropperSettings.aspectRatio;
168
- const minHeightToContain = originalSize.height / cropperSettings.aspectRatio;
169
- return {
170
- width: Math.max(originalSize.height, minWidthToContain),
171
- height: Math.max(originalSize.width, minHeightToContain)
172
- };
173
- }
174
- else {
175
- const minWidthToContain = originalSize.height * cropperSettings.aspectRatio;
176
- const minHeightToContain = originalSize.width / cropperSettings.aspectRatio;
177
- return {
178
- width: Math.max(originalSize.width, minWidthToContain),
179
- height: Math.max(originalSize.height, minHeightToContain)
180
- };
181
- }
182
- }
183
- if (canvasRotation % 2) {
184
- return {
185
- height: originalSize.width,
186
- width: originalSize.height
187
- };
188
- }
189
- return {
190
- width: originalSize.width,
191
- height: originalSize.height
192
- };
193
- }
194
- }
195
- LoadImageService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: LoadImageService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
196
- LoadImageService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: LoadImageService, providedIn: 'root' });
197
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: LoadImageService, decorators: [{
198
- type: Injectable,
199
- args: [{ providedIn: 'root' }]
200
- }] });
201
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"load-image.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-image-cropper/src/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 LoadImageArrayBuffer {\n  originalImage: HTMLImageElement;\n  originalArrayBuffer: ArrayBufferLike;\n  originalObjectUrl: string;\n  originalImageSize?: { width: number; height: number; } | null;\n}\n\n@Injectable({providedIn: 'root'})\nexport class LoadImageService {\n\n  private autoRotateSupported: Promise<boolean> = supportsAutomaticRotation();\n\n  async loadImageFile(file: File, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    const arrayBuffer = await file.arrayBuffer();\n    return await this.checkImageTypeAndLoadImageFromArrayBuffer(arrayBuffer, file.type, cropperSettings);\n  }\n\n  private checkImageTypeAndLoadImageFromArrayBuffer(arrayBuffer: ArrayBufferLike, imageType: string, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    if (!this.isValidImageType(imageType)) {\n      return Promise.reject(new Error('Invalid image type'));\n    }\n    return this.loadImageFromArrayBuffer(arrayBuffer, cropperSettings, imageType);\n  }\n\n  private isValidImageType(type: string): boolean {\n    return /image\\/(png|jpg|jpeg|bmp|gif|tiff|svg|webp|x-icon|vnd.microsoft.icon)/.test(type);\n  }\n\n  async loadImageFromURL(url: string, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    const res = await fetch(url);\n    const blob = await res.blob();\n    const buffer = await blob.arrayBuffer();\n    return await this.loadImageFromArrayBuffer(buffer, cropperSettings, blob.type);\n  }\n\n  loadBase64Image(imageBase64: string, cropperSettings: CropperSettings): Promise<LoadedImage> {\n    const arrayBuffer = this.base64ToArrayBuffer(imageBase64);\n    return this.loadImageFromArrayBuffer(arrayBuffer, cropperSettings);\n  }\n\n  private base64ToArrayBuffer(imageBase64: string): ArrayBufferLike {\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\n  private async loadImageFromArrayBuffer(arrayBuffer: ArrayBufferLike, cropperSettings: CropperSettings, imageType?: string): Promise<LoadedImage> {\n    const res = await new Promise<LoadImageArrayBuffer>(async (resolve, reject) => {\n      try {\n        const blob = new Blob([arrayBuffer], imageType ? {type: imageType} : undefined);\n        const objectUrl = URL.createObjectURL(blob);\n        const originalImage = new Image();\n        const isSvg = imageType === 'image/svg+xml';\n        const originalImageSize = isSvg ? await this.getSvgImageSize(blob) : undefined;\n        originalImage.onload = () => resolve({\n          originalImage,\n          originalImageSize,\n          originalObjectUrl: objectUrl,\n          originalArrayBuffer: arrayBuffer\n        });\n        originalImage.onerror = reject;\n        originalImage.src = objectUrl;\n      } catch (e) {\n        reject(e);\n      }\n    });\n    return await this.transformImageFromArrayBuffer(res, cropperSettings, res.originalImageSize != null);\n  }\n\n  private async getSvgImageSize(blob: Blob): Promise<{ width: number; height: number; } | null> {\n    const parser = new DOMParser();\n    const doc = parser.parseFromString(await blob.text(), 'image/svg+xml');\n    const svgElement = doc.querySelector('svg');\n    if (!svgElement) {\n      throw Error('Failed to parse SVG image');\n    }\n    const widthAttr = svgElement.getAttribute('width');\n    const heightAttr = svgElement.getAttribute('height');\n    if (widthAttr && heightAttr) {\n      return null;\n    }\n    const viewBoxAttr = svgElement.getAttribute('viewBox')\n      || svgElement.getAttribute('viewbox');\n    if (viewBoxAttr) {\n      const viewBox = viewBoxAttr.split(' ');\n      return {\n        width: +viewBox[2],\n        height: +viewBox[3]\n      };\n    }\n    throw Error('Failed to load SVG image. SVG must have width + height or viewBox definition.');\n  }\n\n  private async transformImageFromArrayBuffer(res: LoadImageArrayBuffer, cropperSettings: CropperSettings, forceTransform = false): Promise<LoadedImage> {\n    const autoRotate = await this.autoRotateSupported;\n    const exifTransform = getTransformationsFromExifData(autoRotate ? -1 : res.originalArrayBuffer);\n    if (!res.originalImage || !res.originalImage.complete) {\n      return Promise.reject(new Error('No image loaded'));\n    }\n    const loadedImage = {\n      original: {\n        objectUrl: res.originalObjectUrl,\n        image: res.originalImage,\n        size: res.originalImageSize ?? {\n          width: res.originalImage.naturalWidth,\n          height: res.originalImage.naturalHeight\n        }\n      },\n      exifTransform\n    };\n    return this.transformLoadedImage(loadedImage, cropperSettings, forceTransform);\n  }\n\n  async transformLoadedImage(loadedImage: Partial<LoadedImage>, cropperSettings: CropperSettings, forceTransform = false): Promise<LoadedImage> {\n    const canvasRotation = cropperSettings.canvasRotation + loadedImage.exifTransform!.rotate;\n    const originalSize = loadedImage.original!.size;\n    if (!forceTransform && canvasRotation === 0 && !loadedImage.exifTransform!.flip && !cropperSettings.containWithinAspectRatio) {\n      return {\n        original: {\n          objectUrl: loadedImage.original!.objectUrl,\n          image: loadedImage.original!.image,\n          size: {...originalSize}\n        },\n        transformed: {\n          objectUrl: loadedImage.original!.objectUrl,\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 blob = await new Promise<Blob | null>(resolve => canvas.toBlob(resolve, cropperSettings.format));\n    if (!blob) {\n      throw new Error('Failed to get Blob for transformed image.');\n    }\n    const objectUrl = URL.createObjectURL(blob);\n    const transformedImage = await this.loadImageFromObjectUrl(objectUrl);\n    return {\n      original: {\n        objectUrl: loadedImage.original!.objectUrl,\n        image: loadedImage.original!.image,\n        size: {...originalSize}\n      },\n      transformed: {\n        objectUrl: objectUrl,\n        image: transformedImage,\n        size: {\n          width: transformedImage.width,\n          height: transformedImage.height\n        }\n      },\n      exifTransform: loadedImage.exifTransform!\n    };\n  }\n\n  private loadImageFromObjectUrl(objectUrl: 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 = objectUrl;\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,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFtbWVyLnV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LWltYWdlLWNyb3BwZXIvc3JjL2xpYi91dGlscy9oYW1tZXIudXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIEhhbW1lclN0YXRpYyA9IG5ldyhlbGVtZW50OiBIVE1MRWxlbWVudCB8IFNWR0VsZW1lbnQsIG9wdGlvbnM/OiBhbnkpID0+IEhhbW1lck1hbmFnZXI7XG5cbi8qKiBAZG9jcy1wcml2YXRlICovXG5leHBvcnQgaW50ZXJmYWNlIEhhbW1lck1hbmFnZXIge1xuICBnZXQoZXZlbnROYW1lOiBzdHJpbmcpOiBIYW1tZXJNYW5hZ2VyO1xuXG4gIHNldChvcHRpb25zOiBhbnkpOiBIYW1tZXJNYW5hZ2VyO1xuXG4gIG9uKGV2ZW50TmFtZTogc3RyaW5nLCBoYW5kbGVyOiAoZXY6IGFueSkgPT4gYW55KTogdm9pZDtcbn1cbiJdfQ==
@@ -1,2 +0,0 @@
1
- export * from './index';
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL25neC1pbWFnZS1jcm9wcGVyL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9pbmRleCc7XG4iXX0=