ngx-image-cropper 9.1.2 → 9.1.3
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.
- package/esm2022/lib/component/cropper.state.mjs +11 -2
- package/esm2022/lib/component/image-cropper.component.mjs +3 -3
- package/esm2022/lib/interfaces/crop-input.interface.mjs +2 -0
- package/esm2022/lib/interfaces/cropper-options.interface.mjs +1 -1
- package/esm2022/lib/interfaces/index.mjs +1 -1
- package/esm2022/lib/services/crop.service.mjs +51 -54
- package/fesm2022/ngx-image-cropper.mjs +62 -56
- package/fesm2022/ngx-image-cropper.mjs.map +1 -1
- package/lib/component/cropper.state.d.ts +2 -2
- package/lib/interfaces/crop-input.interface.d.ts +23 -0
- package/lib/interfaces/cropper-options.interface.d.ts +1 -1
- package/lib/interfaces/index.d.ts +1 -0
- package/lib/services/crop.service.d.ts +12 -6
- package/package.json +1 -1
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import { resizeCanvas } from '../utils/resize.utils';
|
|
2
2
|
import { percentage } from '../utils/percentage.utils';
|
|
3
3
|
export class CropService {
|
|
4
|
-
crop(
|
|
5
|
-
const imagePosition = this.getImagePosition(
|
|
4
|
+
crop(input, output) {
|
|
5
|
+
const imagePosition = this.getImagePosition(input);
|
|
6
6
|
const width = imagePosition.x2 - imagePosition.x1;
|
|
7
7
|
const height = imagePosition.y2 - imagePosition.y1;
|
|
8
8
|
const cropCanvas = document.createElement('canvas');
|
|
@@ -12,77 +12,76 @@ export class CropService {
|
|
|
12
12
|
if (!ctx) {
|
|
13
13
|
return null;
|
|
14
14
|
}
|
|
15
|
-
if (
|
|
16
|
-
ctx.fillStyle =
|
|
15
|
+
if (input.options?.backgroundColor != null) {
|
|
16
|
+
ctx.fillStyle = input.options.backgroundColor;
|
|
17
17
|
ctx.fillRect(0, 0, width, height);
|
|
18
18
|
}
|
|
19
|
-
const scaleX = (
|
|
20
|
-
const scaleY = (
|
|
21
|
-
const { translateH, translateV } = this.getCanvasTranslate(
|
|
22
|
-
const transformedImage =
|
|
19
|
+
const scaleX = (input.transform?.scale || 1) * (input.transform?.flipH ? -1 : 1);
|
|
20
|
+
const scaleY = (input.transform?.scale || 1) * (input.transform?.flipV ? -1 : 1);
|
|
21
|
+
const { translateH, translateV } = this.getCanvasTranslate(input);
|
|
22
|
+
const transformedImage = input.loadedImage.transformed;
|
|
23
23
|
ctx.setTransform(scaleX, 0, 0, scaleY, transformedImage.size.width / 2 + translateH, transformedImage.size.height / 2 + translateV);
|
|
24
24
|
ctx.translate(-imagePosition.x1 / scaleX, -imagePosition.y1 / scaleY);
|
|
25
|
-
ctx.rotate((
|
|
25
|
+
ctx.rotate((input.transform?.rotate || 0) * Math.PI / 180);
|
|
26
26
|
ctx.drawImage(transformedImage.image, -transformedImage.size.width / 2, -transformedImage.size.height / 2);
|
|
27
27
|
const result = {
|
|
28
28
|
width, height,
|
|
29
29
|
imagePosition,
|
|
30
|
-
cropperPosition: { ...
|
|
30
|
+
cropperPosition: { ...input.cropper }
|
|
31
31
|
};
|
|
32
|
-
if (
|
|
33
|
-
result.offsetImagePosition = this.getOffsetImagePosition(
|
|
32
|
+
if (input.options?.containWithinAspectRatio) {
|
|
33
|
+
result.offsetImagePosition = this.getOffsetImagePosition(input);
|
|
34
34
|
}
|
|
35
|
-
const resizeRatio = this.getResizeRatio(width, height,
|
|
35
|
+
const resizeRatio = this.getResizeRatio(width, height, input.options);
|
|
36
36
|
if (resizeRatio !== 1) {
|
|
37
37
|
result.width = Math.round(width * resizeRatio);
|
|
38
|
-
result.height =
|
|
39
|
-
? Math.round(result.width /
|
|
38
|
+
result.height = input.options?.maintainAspectRatio
|
|
39
|
+
? Math.round(result.width / (input.options?.aspectRatio ?? 1))
|
|
40
40
|
: Math.round(height * resizeRatio);
|
|
41
41
|
resizeCanvas(cropCanvas, result.width, result.height);
|
|
42
42
|
}
|
|
43
43
|
if (output === 'blob') {
|
|
44
|
-
return this.cropToBlob(result, cropCanvas,
|
|
44
|
+
return this.cropToBlob(result, cropCanvas, input);
|
|
45
45
|
}
|
|
46
46
|
else {
|
|
47
|
-
result.base64 = cropCanvas.toDataURL('image/' +
|
|
47
|
+
result.base64 = cropCanvas.toDataURL('image/' + (input.options?.format ?? 'png'), this.getQuality(input.options));
|
|
48
48
|
return result;
|
|
49
49
|
}
|
|
50
50
|
}
|
|
51
|
-
async cropToBlob(output, cropCanvas,
|
|
52
|
-
output.blob = await new Promise(resolve => cropCanvas.toBlob(resolve, 'image/' +
|
|
51
|
+
async cropToBlob(output, cropCanvas, input) {
|
|
52
|
+
output.blob = await new Promise(resolve => cropCanvas.toBlob(resolve, 'image/' + (input.options?.format ?? 'png'), this.getQuality(input.options)));
|
|
53
53
|
if (output.blob) {
|
|
54
54
|
output.objectUrl = URL.createObjectURL(output.blob);
|
|
55
55
|
}
|
|
56
56
|
return output;
|
|
57
57
|
}
|
|
58
|
-
getCanvasTranslate(
|
|
59
|
-
if (
|
|
60
|
-
const ratio = this.getRatio(
|
|
58
|
+
getCanvasTranslate(input) {
|
|
59
|
+
if (input.transform?.translateUnit === 'px') {
|
|
60
|
+
const ratio = this.getRatio(input);
|
|
61
61
|
return {
|
|
62
|
-
translateH: (
|
|
63
|
-
translateV: (
|
|
62
|
+
translateH: (input.transform?.translateH || 0) * ratio,
|
|
63
|
+
translateV: (input.transform?.translateV || 0) * ratio
|
|
64
64
|
};
|
|
65
65
|
}
|
|
66
66
|
else {
|
|
67
67
|
return {
|
|
68
|
-
translateH:
|
|
69
|
-
translateV:
|
|
68
|
+
translateH: input.transform?.translateH ? percentage(input.transform.translateH, input.loadedImage.transformed.size.width) : 0,
|
|
69
|
+
translateV: input.transform?.translateV ? percentage(input.transform.translateV, input.loadedImage.transformed.size.height) : 0
|
|
70
70
|
};
|
|
71
71
|
}
|
|
72
72
|
}
|
|
73
|
-
getRatio(
|
|
74
|
-
return
|
|
73
|
+
getRatio(input) {
|
|
74
|
+
return input.loadedImage.transformed.size.width / input.maxSize.width;
|
|
75
75
|
}
|
|
76
76
|
getImagePosition(cropperState) {
|
|
77
77
|
const ratio = this.getRatio(cropperState);
|
|
78
|
-
const cropper = cropperState.cropper();
|
|
79
78
|
const out = {
|
|
80
|
-
x1: Math.round(cropper.x1 * ratio),
|
|
81
|
-
y1: Math.round(cropper.y1 * ratio),
|
|
82
|
-
x2: Math.round(cropper.x2 * ratio),
|
|
83
|
-
y2: Math.round(cropper.y2 * ratio)
|
|
79
|
+
x1: Math.round(cropperState.cropper.x1 * ratio),
|
|
80
|
+
y1: Math.round(cropperState.cropper.y1 * ratio),
|
|
81
|
+
x2: Math.round(cropperState.cropper.x2 * ratio),
|
|
82
|
+
y2: Math.round(cropperState.cropper.y2 * ratio)
|
|
84
83
|
};
|
|
85
|
-
if (!cropperState.options
|
|
84
|
+
if (!cropperState.options?.containWithinAspectRatio) {
|
|
86
85
|
out.x1 = Math.max(out.x1, 0);
|
|
87
86
|
out.y1 = Math.max(out.y1, 0);
|
|
88
87
|
out.x2 = Math.min(out.x2, cropperState.loadedImage.transformed.size.width);
|
|
@@ -90,52 +89,50 @@ export class CropService {
|
|
|
90
89
|
}
|
|
91
90
|
return out;
|
|
92
91
|
}
|
|
93
|
-
getOffsetImagePosition(
|
|
94
|
-
const canvasRotation =
|
|
95
|
-
const ratio = this.getRatio(
|
|
92
|
+
getOffsetImagePosition(input) {
|
|
93
|
+
const canvasRotation = (input.options?.canvasRotation ?? 0) + input.loadedImage.exifTransform.rotate;
|
|
94
|
+
const ratio = this.getRatio(input);
|
|
96
95
|
let offsetX;
|
|
97
96
|
let offsetY;
|
|
98
97
|
if (canvasRotation % 2) {
|
|
99
|
-
offsetX = (
|
|
100
|
-
offsetY = (
|
|
98
|
+
offsetX = (input.loadedImage.transformed.size.width - input.loadedImage.original.size.height) / 2;
|
|
99
|
+
offsetY = (input.loadedImage.transformed.size.height - input.loadedImage.original.size.width) / 2;
|
|
101
100
|
}
|
|
102
101
|
else {
|
|
103
|
-
offsetX = (
|
|
104
|
-
offsetY = (
|
|
102
|
+
offsetX = (input.loadedImage.transformed.size.width - input.loadedImage.original.size.width) / 2;
|
|
103
|
+
offsetY = (input.loadedImage.transformed.size.height - input.loadedImage.original.size.height) / 2;
|
|
105
104
|
}
|
|
106
|
-
const cropper =
|
|
105
|
+
const cropper = input.cropper;
|
|
107
106
|
const out = {
|
|
108
107
|
x1: Math.round(cropper.x1 * ratio) - offsetX,
|
|
109
108
|
y1: Math.round(cropper.y1 * ratio) - offsetY,
|
|
110
109
|
x2: Math.round(cropper.x2 * ratio) - offsetX,
|
|
111
110
|
y2: Math.round(cropper.y2 * ratio) - offsetY
|
|
112
111
|
};
|
|
113
|
-
if (!
|
|
112
|
+
if (!input.options?.containWithinAspectRatio) {
|
|
114
113
|
out.x1 = Math.max(out.x1, 0);
|
|
115
114
|
out.y1 = Math.max(out.y1, 0);
|
|
116
|
-
out.x2 = Math.min(out.x2,
|
|
117
|
-
out.y2 = Math.min(out.y2,
|
|
115
|
+
out.x2 = Math.min(out.x2, input.loadedImage.transformed.size.width);
|
|
116
|
+
out.y2 = Math.min(out.y2, input.loadedImage.transformed.size.height);
|
|
118
117
|
}
|
|
119
118
|
return out;
|
|
120
119
|
}
|
|
121
120
|
getResizeRatio(width, height, options) {
|
|
122
|
-
const ratioWidth = options.resizeToWidth / width;
|
|
123
|
-
const ratioHeight = options.resizeToHeight / height;
|
|
124
121
|
const ratios = new Array();
|
|
125
|
-
if (options.resizeToWidth > 0) {
|
|
126
|
-
ratios.push(
|
|
122
|
+
if (options?.resizeToWidth && options.resizeToWidth > 0) {
|
|
123
|
+
ratios.push(options.resizeToWidth / width);
|
|
127
124
|
}
|
|
128
|
-
if (options.resizeToHeight > 0) {
|
|
129
|
-
ratios.push(
|
|
125
|
+
if (options?.resizeToHeight && options.resizeToHeight > 0) {
|
|
126
|
+
ratios.push(options.resizeToHeight / height);
|
|
130
127
|
}
|
|
131
128
|
const result = ratios.length === 0 ? 1 : Math.min(...ratios);
|
|
132
|
-
if (result > 1 && !options
|
|
129
|
+
if (result > 1 && !options?.onlyScaleDown) {
|
|
133
130
|
return result;
|
|
134
131
|
}
|
|
135
132
|
return Math.min(result, 1);
|
|
136
133
|
}
|
|
137
134
|
getQuality(options) {
|
|
138
|
-
return Math.min(1, Math.max(0, options
|
|
135
|
+
return Math.min(1, Math.max(0, options?.imageQuality ?? 92 / 100));
|
|
139
136
|
}
|
|
140
137
|
}
|
|
141
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crop.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-image-cropper/src/lib/services/crop.service.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,UAAU,EAAE,MAAM,2BAA2B,CAAC;AAGvD,MAAM,OAAO,WAAW;IAItB,IAAI,CAAC,YAA0B,EAAE,MAAkB;QACjD,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC,YAAY,CAAC,CAAC;QAC1D,MAAM,KAAK,GAAG,aAAa,CAAC,EAAE,GAAG,aAAa,CAAC,EAAE,CAAC;QAClD,MAAM,MAAM,GAAG,aAAa,CAAC,EAAE,GAAG,aAAa,CAAC,EAAE,CAAC;QACnD,MAAM,UAAU,GAAG,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAsB,CAAC;QACzE,UAAU,CAAC,KAAK,GAAG,KAAK,CAAC;QACzB,UAAU,CAAC,MAAM,GAAG,MAAM,CAAC;QAE3B,MAAM,GAAG,GAAG,UAAU,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACxC,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,OAAO,IAAI,CAAC;QACd,CAAC;QACD,IAAI,YAAY,CAAC,OAAO,CAAC,eAAe,IAAI,IAAI,EAAE,CAAC;YACjD,GAAG,CAAC,SAAS,GAAG,YAAY,CAAC,OAAO,CAAC,eAAe,CAAC;YACrD,GAAG,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;QACpC,CAAC;QAED,MAAM,MAAM,GAAG,CAAC,YAAY,CAAC,SAAS,CAAC,KAAK,IAAI,CAAC,CAAC,GAAG,CAAC,YAAY,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC7F,MAAM,MAAM,GAAG,CAAC,YAAY,CAAC,SAAS,CAAC,KAAK,IAAI,CAAC,CAAC,GAAG,CAAC,YAAY,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC7F,MAAM,EAAC,UAAU,EAAE,UAAU,EAAC,GAAG,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;QAEvE,MAAM,gBAAgB,GAAG,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC;QAC/D,GAAG,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC,EAAE,CAAC,EAAE,MAAM,EAAE,gBAAgB,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,GAAG,UAAU,EAAE,gBAAgB,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,GAAG,UAAU,CAAC,CAAC;QACpI,GAAG,CAAC,SAAS,CAAC,CAAC,aAAa,CAAC,EAAE,GAAG,MAAM,EAAE,CAAC,aAAa,CAAC,EAAE,GAAG,MAAM,CAAC,CAAC;QACtE,GAAG,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,SAAS,CAAC,MAAM,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,EAAE,GAAG,GAAG,CAAC,CAAC;QAEjE,GAAG,CAAC,SAAS,CACX,gBAAgB,CAAC,KAAK,EACtB,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,EAChC,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAClC,CAAC;QAEF,MAAM,MAAM,GAAsB;YAChC,KAAK,EAAE,MAAM;YACb,aAAa;YACb,eAAe,EAAE,EAAC,GAAG,YAAY,CAAC,OAAO,EAAE,EAAC;SAC7C,CAAC;QACF,IAAI,YAAY,CAAC,OAAO,CAAC,wBAAwB,EAAE,CAAC;YAClD,MAAM,CAAC,mBAAmB,GAAG,IAAI,CAAC,sBAAsB,CAAC,YAAY,CAAC,CAAC;QACzE,CAAC;QACD,MAAM,WAAW,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,MAAM,EAAE,YAAY,CAAC,OAAO,CAAC,CAAC;QAC7E,IAAI,WAAW,KAAK,CAAC,EAAE,CAAC;YACtB,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,WAAW,CAAC,CAAC;YAC/C,MAAM,CAAC,MAAM,GAAG,YAAY,CAAC,OAAO,CAAC,mBAAmB;gBACtD,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,GAAG,YAAY,CAAC,OAAO,CAAC,WAAW,CAAC;gBAC7D,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,WAAW,CAAC,CAAC;YACrC,YAAY,CAAC,UAAU,EAAE,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;QACxD,CAAC;QACD,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;YACtB,OAAO,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,UAAU,EAAE,YAAY,CAAC,CAAC;QAC3D,CAAC;aAAM,CAAC;YACN,MAAM,CAAC,MAAM,GAAG,UAAU,CAAC,SAAS,CAAC,QAAQ,GAAG,YAAY,CAAC,OAAO,CAAC,MAAM,EAAE,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;YACpH,OAAO,MAAM,CAAC;QAChB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,UAAU,CAAC,MAAyB,EAAE,UAA6B,EAAE,YAA0B;QAC3G,MAAM,CAAC,IAAI,GAAG,MAAM,IAAI,OAAO,CAAc,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,OAAO,EAAE,QAAQ,GAAG,YAAY,CAAC,OAAO,CAAC,MAAM,EAAE,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QACnK,IAAI,MAAM,CAAC,IAAI,EAAE,CAAC;YAChB,MAAM,CAAC,SAAS,GAAG,GAAG,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACtD,CAAC;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAEO,kBAAkB,CAAC,YAA0B;QACnD,IAAI,YAAY,CAAC,SAAS,CAAC,aAAa,KAAK,IAAI,EAAE,CAAC;YAClD,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;YAC1C,OAAO;gBACL,UAAU,EAAE,CAAC,YAAY,CAAC,SAAS,CAAC,UAAU,IAAI,CAAC,CAAC,GAAG,KAAK;gBAC5D,UAAU,EAAE,CAAC,YAAY,CAAC,SAAS,CAAC,UAAU,IAAI,CAAC,CAAC,GAAG,KAAK;aAC7D,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,OAAO;gBACL,UAAU,EAAE,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,YAAY,CAAC,SAAS,CAAC,UAAU,EAAE,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;gBACnJ,UAAU,EAAE,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,YAAY,CAAC,SAAS,CAAC,UAAU,EAAE,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;aACrJ,CAAC;QACJ,CAAC;IACH,CAAC;IAEO,QAAQ,CAAC,YAA0B;QACzC,OAAO,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,GAAG,YAAY,CAAC,OAAO,EAAE,CAAC,KAAK,CAAC;IACzF,CAAC;IAEO,gBAAgB,CAAC,YAA0B;QACjD,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;QAC1C,MAAM,OAAO,GAAG,YAAY,CAAC,OAAO,EAAE,CAAC;QACvC,MAAM,GAAG,GAAoB;YAC3B,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC;YAClC,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC;YAClC,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC;YAClC,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC;SACnC,CAAC;QAEF,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,wBAAwB,EAAE,CAAC;YACnD,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;YAC7B,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;YAC7B,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5E,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC/E,CAAC;QAED,OAAO,GAAG,CAAC;IACb,CAAC;IAEO,sBAAsB,CAAC,YAA0B;QACvD,MAAM,cAAc,GAAG,YAAY,CAAC,OAAO,CAAC,cAAc,GAAG,YAAY,CAAC,WAAY,CAAC,aAAa,CAAC,MAAM,CAAC;QAC5G,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;QAC1C,IAAI,OAAe,CAAC;QACpB,IAAI,OAAe,CAAC;QAEpB,IAAI,cAAc,GAAG,CAAC,EAAE,CAAC;YACvB,OAAO,GAAG,CAAC,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,GAAG,YAAY,CAAC,WAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YAClH,OAAO,GAAG,CAAC,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC,WAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACpH,CAAC;aAAM,CAAC;YACN,OAAO,GAAG,CAAC,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,GAAG,YAAY,CAAC,WAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACjH,OAAO,GAAG,CAAC,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC,WAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACrH,CAAC;QAED,MAAM,OAAO,GAAG,YAAY,CAAC,OAAO,EAAE,CAAC;QACvC,MAAM,GAAG,GAAoB;YAC3B,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC,GAAG,OAAO;YAC5C,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC,GAAG,OAAO;YAC5C,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC,GAAG,OAAO;YAC5C,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC,GAAG,OAAO;SAC7C,CAAC;QAEF,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,wBAAwB,EAAE,CAAC;YACnD,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;YAC7B,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;YAC7B,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5E,GAAG,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,YAAY,CAAC,WAAY,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC/E,CAAC;QAED,OAAO,GAAG,CAAC;IACb,CAAC;IAED,cAAc,CAAC,KAAa,EAAE,MAAc,EAAE,OAAuB;QACnE,MAAM,UAAU,GAAG,OAAO,CAAC,aAAa,GAAG,KAAK,CAAC;QACjD,MAAM,WAAW,GAAG,OAAO,CAAC,cAAc,GAAG,MAAM,CAAC;QACpD,MAAM,MAAM,GAAG,IAAI,KAAK,EAAU,CAAC;QAEnC,IAAI,OAAO,CAAC,aAAa,GAAG,CAAC,EAAE,CAAC;YAC9B,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC1B,CAAC;QACD,IAAI,OAAO,CAAC,cAAc,GAAG,CAAC,EAAE,CAAC;YAC/B,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC3B,CAAC;QAED,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,CAAC;QAE7D,IAAI,MAAM,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,CAAC;YACzC,OAAO,MAAM,CAAC;QAChB,CAAC;QACD,OAAO,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAC7B,CAAC;IAED,UAAU,CAAC,OAAuB;QAChC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,CAAC,YAAY,GAAG,GAAG,CAAC,CAAC,CAAC;IAC9D,CAAC;CACF","sourcesContent":["import { CropperOptions, CropperPosition, ImageCroppedEvent } from '../interfaces';\nimport { CropperState } from '../component/cropper.state';\nimport { resizeCanvas } from '../utils/resize.utils';\nimport { percentage } from '../utils/percentage.utils';\nimport { OutputType } from '../interfaces/cropper-options.interface';\n\nexport class CropService {\n\n  crop(cropperState: CropperState, output: 'blob'): Promise<ImageCroppedEvent> | null;\n  crop(cropperState: CropperState, output: 'base64'): ImageCroppedEvent | null;\n  crop(cropperState: CropperState, output: OutputType): Promise<ImageCroppedEvent> | ImageCroppedEvent | null {\n    const imagePosition = this.getImagePosition(cropperState);\n    const width = imagePosition.x2 - imagePosition.x1;\n    const height = imagePosition.y2 - imagePosition.y1;\n    const cropCanvas = document.createElement('canvas') as HTMLCanvasElement;\n    cropCanvas.width = width;\n    cropCanvas.height = height;\n\n    const ctx = cropCanvas.getContext('2d');\n    if (!ctx) {\n      return null;\n    }\n    if (cropperState.options.backgroundColor != null) {\n      ctx.fillStyle = cropperState.options.backgroundColor;\n      ctx.fillRect(0, 0, width, height);\n    }\n\n    const scaleX = (cropperState.transform.scale || 1) * (cropperState.transform.flipH ? -1 : 1);\n    const scaleY = (cropperState.transform.scale || 1) * (cropperState.transform.flipV ? -1 : 1);\n    const {translateH, translateV} = this.getCanvasTranslate(cropperState);\n\n    const transformedImage = cropperState.loadedImage!.transformed;\n    ctx.setTransform(scaleX, 0, 0, scaleY, transformedImage.size.width / 2 + translateH, transformedImage.size.height / 2 + translateV);\n    ctx.translate(-imagePosition.x1 / scaleX, -imagePosition.y1 / scaleY);\n    ctx.rotate((cropperState.transform.rotate || 0) * Math.PI / 180);\n\n    ctx.drawImage(\n      transformedImage.image,\n      -transformedImage.size.width / 2,\n      -transformedImage.size.height / 2\n    );\n\n    const result: ImageCroppedEvent = {\n      width, height,\n      imagePosition,\n      cropperPosition: {...cropperState.cropper()}\n    };\n    if (cropperState.options.containWithinAspectRatio) {\n      result.offsetImagePosition = this.getOffsetImagePosition(cropperState);\n    }\n    const resizeRatio = this.getResizeRatio(width, height, cropperState.options);\n    if (resizeRatio !== 1) {\n      result.width = Math.round(width * resizeRatio);\n      result.height = cropperState.options.maintainAspectRatio\n        ? Math.round(result.width / cropperState.options.aspectRatio)\n        : Math.round(height * resizeRatio);\n      resizeCanvas(cropCanvas, result.width, result.height);\n    }\n    if (output === 'blob') {\n      return this.cropToBlob(result, cropCanvas, cropperState);\n    } else {\n      result.base64 = cropCanvas.toDataURL('image/' + cropperState.options.format, this.getQuality(cropperState.options));\n      return result;\n    }\n  }\n\n  private async cropToBlob(output: ImageCroppedEvent, cropCanvas: HTMLCanvasElement, cropperState: CropperState): Promise<ImageCroppedEvent> {\n    output.blob = await new Promise<Blob | null>(resolve => cropCanvas.toBlob(resolve, 'image/' + cropperState.options.format, this.getQuality(cropperState.options)));\n    if (output.blob) {\n      output.objectUrl = URL.createObjectURL(output.blob);\n    }\n    return output;\n  }\n\n  private getCanvasTranslate(cropperState: CropperState): { translateH: number, translateV: number } {\n    if (cropperState.transform.translateUnit === 'px') {\n      const ratio = this.getRatio(cropperState);\n      return {\n        translateH: (cropperState.transform.translateH || 0) * ratio,\n        translateV: (cropperState.transform.translateV || 0) * ratio\n      };\n    } else {\n      return {\n        translateH: cropperState.transform.translateH ? percentage(cropperState.transform.translateH, cropperState.loadedImage!.transformed.size.width) : 0,\n        translateV: cropperState.transform.translateV ? percentage(cropperState.transform.translateV, cropperState.loadedImage!.transformed.size.height) : 0\n      };\n    }\n  }\n\n  private getRatio(cropperState: CropperState): number {\n    return cropperState.loadedImage!.transformed.size.width / cropperState.maxSize().width;\n  }\n\n  private getImagePosition(cropperState: CropperState): CropperPosition {\n    const ratio = this.getRatio(cropperState);\n    const cropper = cropperState.cropper();\n    const out: CropperPosition = {\n      x1: Math.round(cropper.x1 * ratio),\n      y1: Math.round(cropper.y1 * ratio),\n      x2: Math.round(cropper.x2 * ratio),\n      y2: Math.round(cropper.y2 * ratio)\n    };\n\n    if (!cropperState.options.containWithinAspectRatio) {\n      out.x1 = Math.max(out.x1, 0);\n      out.y1 = Math.max(out.y1, 0);\n      out.x2 = Math.min(out.x2, cropperState.loadedImage!.transformed.size.width);\n      out.y2 = Math.min(out.y2, cropperState.loadedImage!.transformed.size.height);\n    }\n\n    return out;\n  }\n\n  private getOffsetImagePosition(cropperState: CropperState): CropperPosition {\n    const canvasRotation = cropperState.options.canvasRotation + cropperState.loadedImage!.exifTransform.rotate;\n    const ratio = this.getRatio(cropperState);\n    let offsetX: number;\n    let offsetY: number;\n\n    if (canvasRotation % 2) {\n      offsetX = (cropperState.loadedImage!.transformed.size.width - cropperState.loadedImage!.original.size.height) / 2;\n      offsetY = (cropperState.loadedImage!.transformed.size.height - cropperState.loadedImage!.original.size.width) / 2;\n    } else {\n      offsetX = (cropperState.loadedImage!.transformed.size.width - cropperState.loadedImage!.original.size.width) / 2;\n      offsetY = (cropperState.loadedImage!.transformed.size.height - cropperState.loadedImage!.original.size.height) / 2;\n    }\n\n    const cropper = cropperState.cropper();\n    const out: CropperPosition = {\n      x1: Math.round(cropper.x1 * ratio) - offsetX,\n      y1: Math.round(cropper.y1 * ratio) - offsetY,\n      x2: Math.round(cropper.x2 * ratio) - offsetX,\n      y2: Math.round(cropper.y2 * ratio) - offsetY\n    };\n\n    if (!cropperState.options.containWithinAspectRatio) {\n      out.x1 = Math.max(out.x1, 0);\n      out.y1 = Math.max(out.y1, 0);\n      out.x2 = Math.min(out.x2, cropperState.loadedImage!.transformed.size.width);\n      out.y2 = Math.min(out.y2, cropperState.loadedImage!.transformed.size.height);\n    }\n\n    return out;\n  }\n\n  getResizeRatio(width: number, height: number, options: CropperOptions): number {\n    const ratioWidth = options.resizeToWidth / width;\n    const ratioHeight = options.resizeToHeight / height;\n    const ratios = new Array<number>();\n\n    if (options.resizeToWidth > 0) {\n      ratios.push(ratioWidth);\n    }\n    if (options.resizeToHeight > 0) {\n      ratios.push(ratioHeight);\n    }\n\n    const result = ratios.length === 0 ? 1 : Math.min(...ratios);\n\n    if (result > 1 && !options.onlyScaleDown) {\n      return result;\n    }\n    return Math.min(result, 1);\n  }\n\n  getQuality(options: CropperOptions): number {\n    return Math.min(1, Math.max(0, options.imageQuality / 100));\n  }\n}\n"]}
|
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crop.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-image-cropper/src/lib/services/crop.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { CropperPosition, ImageCroppedEvent } from '../interfaces';\nimport { resizeCanvas } from '../utils/resize.utils';\nimport { percentage } from '../utils/percentage.utils';\nimport { OutputType } from '../interfaces/cropper-options.interface';\nimport { CropInput } from '../interfaces/crop-input.interface';\n\nexport class CropService {\n\n  crop(input: CropInput, output: 'blob'): Promise<ImageCroppedEvent> | null;\n  crop(input: CropInput, output: 'base64'): ImageCroppedEvent | null;\n  crop(input: CropInput, output: OutputType): Promise<ImageCroppedEvent> | ImageCroppedEvent | null {\n    const imagePosition = this.getImagePosition(input);\n    const width = imagePosition.x2 - imagePosition.x1;\n    const height = imagePosition.y2 - imagePosition.y1;\n    const cropCanvas = document.createElement('canvas') as HTMLCanvasElement;\n    cropCanvas.width = width;\n    cropCanvas.height = height;\n\n    const ctx = cropCanvas.getContext('2d');\n    if (!ctx) {\n      return null;\n    }\n    if (input.options?.backgroundColor != null) {\n      ctx.fillStyle = input.options.backgroundColor;\n      ctx.fillRect(0, 0, width, height);\n    }\n\n    const scaleX = (input.transform?.scale || 1) * (input.transform?.flipH ? -1 : 1);\n    const scaleY = (input.transform?.scale || 1) * (input.transform?.flipV ? -1 : 1);\n    const {translateH, translateV} = this.getCanvasTranslate(input);\n\n    const transformedImage = input.loadedImage!.transformed;\n    ctx.setTransform(scaleX, 0, 0, scaleY, transformedImage.size.width / 2 + translateH, transformedImage.size.height / 2 + translateV);\n    ctx.translate(-imagePosition.x1 / scaleX, -imagePosition.y1 / scaleY);\n    ctx.rotate((input.transform?.rotate || 0) * Math.PI / 180);\n\n    ctx.drawImage(\n      transformedImage.image,\n      -transformedImage.size.width / 2,\n      -transformedImage.size.height / 2\n    );\n\n    const result: ImageCroppedEvent = {\n      width, height,\n      imagePosition,\n      cropperPosition: {...input.cropper}\n    };\n    if (input.options?.containWithinAspectRatio) {\n      result.offsetImagePosition = this.getOffsetImagePosition(input);\n    }\n    const resizeRatio = this.getResizeRatio(width, height, input.options);\n    if (resizeRatio !== 1) {\n      result.width = Math.round(width * resizeRatio);\n      result.height = input.options?.maintainAspectRatio\n        ? Math.round(result.width / (input.options?.aspectRatio ?? 1))\n        : Math.round(height * resizeRatio);\n      resizeCanvas(cropCanvas, result.width, result.height);\n    }\n    if (output === 'blob') {\n      return this.cropToBlob(result, cropCanvas, input);\n    } else {\n      result.base64 = cropCanvas.toDataURL('image/' + (input.options?.format ?? 'png'), this.getQuality(input.options));\n      return result;\n    }\n  }\n\n  private async cropToBlob(output: ImageCroppedEvent, cropCanvas: HTMLCanvasElement, input: CropInput): Promise<ImageCroppedEvent> {\n    output.blob = await new Promise<Blob | null>(resolve => cropCanvas.toBlob(resolve, 'image/' + (input.options?.format ?? 'png'), this.getQuality(input.options)));\n    if (output.blob) {\n      output.objectUrl = URL.createObjectURL(output.blob);\n    }\n    return output;\n  }\n\n  private getCanvasTranslate(input: CropInput): { translateH: number, translateV: number } {\n    if (input.transform?.translateUnit === 'px') {\n      const ratio = this.getRatio(input);\n      return {\n        translateH: (input.transform?.translateH || 0) * ratio,\n        translateV: (input.transform?.translateV || 0) * ratio\n      };\n    } else {\n      return {\n        translateH: input.transform?.translateH ? percentage(input.transform.translateH, input.loadedImage!.transformed.size.width) : 0,\n        translateV: input.transform?.translateV ? percentage(input.transform.translateV, input.loadedImage!.transformed.size.height) : 0\n      };\n    }\n  }\n\n  private getRatio(input: CropInput): number {\n    return input.loadedImage!.transformed.size.width / input.maxSize.width;\n  }\n\n  private getImagePosition(cropperState: CropInput): CropperPosition {\n    const ratio = this.getRatio(cropperState);\n    const out: CropperPosition = {\n      x1: Math.round(cropperState.cropper.x1 * ratio),\n      y1: Math.round(cropperState.cropper.y1 * ratio),\n      x2: Math.round(cropperState.cropper.x2 * ratio),\n      y2: Math.round(cropperState.cropper.y2 * ratio)\n    };\n\n    if (!cropperState.options?.containWithinAspectRatio) {\n      out.x1 = Math.max(out.x1, 0);\n      out.y1 = Math.max(out.y1, 0);\n      out.x2 = Math.min(out.x2, cropperState.loadedImage!.transformed.size.width);\n      out.y2 = Math.min(out.y2, cropperState.loadedImage!.transformed.size.height);\n    }\n\n    return out;\n  }\n\n  private getOffsetImagePosition(input: CropInput): CropperPosition {\n    const canvasRotation = (input.options?.canvasRotation ?? 0) + input.loadedImage!.exifTransform.rotate;\n    const ratio = this.getRatio(input);\n    let offsetX: number;\n    let offsetY: number;\n\n    if (canvasRotation % 2) {\n      offsetX = (input.loadedImage!.transformed.size.width - input.loadedImage!.original.size.height) / 2;\n      offsetY = (input.loadedImage!.transformed.size.height - input.loadedImage!.original.size.width) / 2;\n    } else {\n      offsetX = (input.loadedImage!.transformed.size.width - input.loadedImage!.original.size.width) / 2;\n      offsetY = (input.loadedImage!.transformed.size.height - input.loadedImage!.original.size.height) / 2;\n    }\n\n    const cropper = input.cropper;\n    const out: CropperPosition = {\n      x1: Math.round(cropper.x1 * ratio) - offsetX,\n      y1: Math.round(cropper.y1 * ratio) - offsetY,\n      x2: Math.round(cropper.x2 * ratio) - offsetX,\n      y2: Math.round(cropper.y2 * ratio) - offsetY\n    };\n\n    if (!input.options?.containWithinAspectRatio) {\n      out.x1 = Math.max(out.x1, 0);\n      out.y1 = Math.max(out.y1, 0);\n      out.x2 = Math.min(out.x2, input.loadedImage!.transformed.size.width);\n      out.y2 = Math.min(out.y2, input.loadedImage!.transformed.size.height);\n    }\n\n    return out;\n  }\n\n  getResizeRatio(width: number, height: number, options?: {\n    resizeToWidth?: number;\n    resizeToHeight?: number;\n    onlyScaleDown?: boolean;\n  }): number {\n    const ratios = new Array<number>();\n    if (options?.resizeToWidth && options.resizeToWidth > 0) {\n      ratios.push(options.resizeToWidth / width);\n    }\n    if (options?.resizeToHeight && options.resizeToHeight > 0) {\n      ratios.push(options.resizeToHeight / height);\n    }\n\n    const result = ratios.length === 0 ? 1 : Math.min(...ratios);\n\n    if (result > 1 && !options?.onlyScaleDown) {\n      return result;\n    }\n    return Math.min(result, 1);\n  }\n\n  getQuality(options?: { imageQuality?: number }): number {\n    return Math.min(1, Math.max(0, options?.imageQuality ?? 92 / 100));\n  }\n}\n"]}
|
|
@@ -268,7 +268,7 @@ class CropperState {
|
|
|
268
268
|
roundCropper: false,
|
|
269
269
|
onlyScaleDown: false,
|
|
270
270
|
imageQuality: 92,
|
|
271
|
-
backgroundColor:
|
|
271
|
+
backgroundColor: undefined,
|
|
272
272
|
containWithinAspectRatio: false,
|
|
273
273
|
hideResizeSquares: false,
|
|
274
274
|
alignImage: 'center',
|
|
@@ -429,6 +429,15 @@ class CropperState {
|
|
|
429
429
|
y2: this.maxSize().height
|
|
430
430
|
};
|
|
431
431
|
}
|
|
432
|
+
toCropInput() {
|
|
433
|
+
return {
|
|
434
|
+
cropper: this.cropper(),
|
|
435
|
+
maxSize: this.maxSize(),
|
|
436
|
+
transform: this.transform,
|
|
437
|
+
loadedImage: this.loadedImage,
|
|
438
|
+
options: { ...this.options }
|
|
439
|
+
};
|
|
440
|
+
}
|
|
432
441
|
}
|
|
433
442
|
|
|
434
443
|
var MoveTypes;
|
|
@@ -519,8 +528,8 @@ function percentage(percent, totalValue) {
|
|
|
519
528
|
}
|
|
520
529
|
|
|
521
530
|
class CropService {
|
|
522
|
-
crop(
|
|
523
|
-
const imagePosition = this.getImagePosition(
|
|
531
|
+
crop(input, output) {
|
|
532
|
+
const imagePosition = this.getImagePosition(input);
|
|
524
533
|
const width = imagePosition.x2 - imagePosition.x1;
|
|
525
534
|
const height = imagePosition.y2 - imagePosition.y1;
|
|
526
535
|
const cropCanvas = document.createElement('canvas');
|
|
@@ -530,77 +539,76 @@ class CropService {
|
|
|
530
539
|
if (!ctx) {
|
|
531
540
|
return null;
|
|
532
541
|
}
|
|
533
|
-
if (
|
|
534
|
-
ctx.fillStyle =
|
|
542
|
+
if (input.options?.backgroundColor != null) {
|
|
543
|
+
ctx.fillStyle = input.options.backgroundColor;
|
|
535
544
|
ctx.fillRect(0, 0, width, height);
|
|
536
545
|
}
|
|
537
|
-
const scaleX = (
|
|
538
|
-
const scaleY = (
|
|
539
|
-
const { translateH, translateV } = this.getCanvasTranslate(
|
|
540
|
-
const transformedImage =
|
|
546
|
+
const scaleX = (input.transform?.scale || 1) * (input.transform?.flipH ? -1 : 1);
|
|
547
|
+
const scaleY = (input.transform?.scale || 1) * (input.transform?.flipV ? -1 : 1);
|
|
548
|
+
const { translateH, translateV } = this.getCanvasTranslate(input);
|
|
549
|
+
const transformedImage = input.loadedImage.transformed;
|
|
541
550
|
ctx.setTransform(scaleX, 0, 0, scaleY, transformedImage.size.width / 2 + translateH, transformedImage.size.height / 2 + translateV);
|
|
542
551
|
ctx.translate(-imagePosition.x1 / scaleX, -imagePosition.y1 / scaleY);
|
|
543
|
-
ctx.rotate((
|
|
552
|
+
ctx.rotate((input.transform?.rotate || 0) * Math.PI / 180);
|
|
544
553
|
ctx.drawImage(transformedImage.image, -transformedImage.size.width / 2, -transformedImage.size.height / 2);
|
|
545
554
|
const result = {
|
|
546
555
|
width, height,
|
|
547
556
|
imagePosition,
|
|
548
|
-
cropperPosition: { ...
|
|
557
|
+
cropperPosition: { ...input.cropper }
|
|
549
558
|
};
|
|
550
|
-
if (
|
|
551
|
-
result.offsetImagePosition = this.getOffsetImagePosition(
|
|
559
|
+
if (input.options?.containWithinAspectRatio) {
|
|
560
|
+
result.offsetImagePosition = this.getOffsetImagePosition(input);
|
|
552
561
|
}
|
|
553
|
-
const resizeRatio = this.getResizeRatio(width, height,
|
|
562
|
+
const resizeRatio = this.getResizeRatio(width, height, input.options);
|
|
554
563
|
if (resizeRatio !== 1) {
|
|
555
564
|
result.width = Math.round(width * resizeRatio);
|
|
556
|
-
result.height =
|
|
557
|
-
? Math.round(result.width /
|
|
565
|
+
result.height = input.options?.maintainAspectRatio
|
|
566
|
+
? Math.round(result.width / (input.options?.aspectRatio ?? 1))
|
|
558
567
|
: Math.round(height * resizeRatio);
|
|
559
568
|
resizeCanvas(cropCanvas, result.width, result.height);
|
|
560
569
|
}
|
|
561
570
|
if (output === 'blob') {
|
|
562
|
-
return this.cropToBlob(result, cropCanvas,
|
|
571
|
+
return this.cropToBlob(result, cropCanvas, input);
|
|
563
572
|
}
|
|
564
573
|
else {
|
|
565
|
-
result.base64 = cropCanvas.toDataURL('image/' +
|
|
574
|
+
result.base64 = cropCanvas.toDataURL('image/' + (input.options?.format ?? 'png'), this.getQuality(input.options));
|
|
566
575
|
return result;
|
|
567
576
|
}
|
|
568
577
|
}
|
|
569
|
-
async cropToBlob(output, cropCanvas,
|
|
570
|
-
output.blob = await new Promise(resolve => cropCanvas.toBlob(resolve, 'image/' +
|
|
578
|
+
async cropToBlob(output, cropCanvas, input) {
|
|
579
|
+
output.blob = await new Promise(resolve => cropCanvas.toBlob(resolve, 'image/' + (input.options?.format ?? 'png'), this.getQuality(input.options)));
|
|
571
580
|
if (output.blob) {
|
|
572
581
|
output.objectUrl = URL.createObjectURL(output.blob);
|
|
573
582
|
}
|
|
574
583
|
return output;
|
|
575
584
|
}
|
|
576
|
-
getCanvasTranslate(
|
|
577
|
-
if (
|
|
578
|
-
const ratio = this.getRatio(
|
|
585
|
+
getCanvasTranslate(input) {
|
|
586
|
+
if (input.transform?.translateUnit === 'px') {
|
|
587
|
+
const ratio = this.getRatio(input);
|
|
579
588
|
return {
|
|
580
|
-
translateH: (
|
|
581
|
-
translateV: (
|
|
589
|
+
translateH: (input.transform?.translateH || 0) * ratio,
|
|
590
|
+
translateV: (input.transform?.translateV || 0) * ratio
|
|
582
591
|
};
|
|
583
592
|
}
|
|
584
593
|
else {
|
|
585
594
|
return {
|
|
586
|
-
translateH:
|
|
587
|
-
translateV:
|
|
595
|
+
translateH: input.transform?.translateH ? percentage(input.transform.translateH, input.loadedImage.transformed.size.width) : 0,
|
|
596
|
+
translateV: input.transform?.translateV ? percentage(input.transform.translateV, input.loadedImage.transformed.size.height) : 0
|
|
588
597
|
};
|
|
589
598
|
}
|
|
590
599
|
}
|
|
591
|
-
getRatio(
|
|
592
|
-
return
|
|
600
|
+
getRatio(input) {
|
|
601
|
+
return input.loadedImage.transformed.size.width / input.maxSize.width;
|
|
593
602
|
}
|
|
594
603
|
getImagePosition(cropperState) {
|
|
595
604
|
const ratio = this.getRatio(cropperState);
|
|
596
|
-
const cropper = cropperState.cropper();
|
|
597
605
|
const out = {
|
|
598
|
-
x1: Math.round(cropper.x1 * ratio),
|
|
599
|
-
y1: Math.round(cropper.y1 * ratio),
|
|
600
|
-
x2: Math.round(cropper.x2 * ratio),
|
|
601
|
-
y2: Math.round(cropper.y2 * ratio)
|
|
606
|
+
x1: Math.round(cropperState.cropper.x1 * ratio),
|
|
607
|
+
y1: Math.round(cropperState.cropper.y1 * ratio),
|
|
608
|
+
x2: Math.round(cropperState.cropper.x2 * ratio),
|
|
609
|
+
y2: Math.round(cropperState.cropper.y2 * ratio)
|
|
602
610
|
};
|
|
603
|
-
if (!cropperState.options
|
|
611
|
+
if (!cropperState.options?.containWithinAspectRatio) {
|
|
604
612
|
out.x1 = Math.max(out.x1, 0);
|
|
605
613
|
out.y1 = Math.max(out.y1, 0);
|
|
606
614
|
out.x2 = Math.min(out.x2, cropperState.loadedImage.transformed.size.width);
|
|
@@ -608,52 +616,50 @@ class CropService {
|
|
|
608
616
|
}
|
|
609
617
|
return out;
|
|
610
618
|
}
|
|
611
|
-
getOffsetImagePosition(
|
|
612
|
-
const canvasRotation =
|
|
613
|
-
const ratio = this.getRatio(
|
|
619
|
+
getOffsetImagePosition(input) {
|
|
620
|
+
const canvasRotation = (input.options?.canvasRotation ?? 0) + input.loadedImage.exifTransform.rotate;
|
|
621
|
+
const ratio = this.getRatio(input);
|
|
614
622
|
let offsetX;
|
|
615
623
|
let offsetY;
|
|
616
624
|
if (canvasRotation % 2) {
|
|
617
|
-
offsetX = (
|
|
618
|
-
offsetY = (
|
|
625
|
+
offsetX = (input.loadedImage.transformed.size.width - input.loadedImage.original.size.height) / 2;
|
|
626
|
+
offsetY = (input.loadedImage.transformed.size.height - input.loadedImage.original.size.width) / 2;
|
|
619
627
|
}
|
|
620
628
|
else {
|
|
621
|
-
offsetX = (
|
|
622
|
-
offsetY = (
|
|
629
|
+
offsetX = (input.loadedImage.transformed.size.width - input.loadedImage.original.size.width) / 2;
|
|
630
|
+
offsetY = (input.loadedImage.transformed.size.height - input.loadedImage.original.size.height) / 2;
|
|
623
631
|
}
|
|
624
|
-
const cropper =
|
|
632
|
+
const cropper = input.cropper;
|
|
625
633
|
const out = {
|
|
626
634
|
x1: Math.round(cropper.x1 * ratio) - offsetX,
|
|
627
635
|
y1: Math.round(cropper.y1 * ratio) - offsetY,
|
|
628
636
|
x2: Math.round(cropper.x2 * ratio) - offsetX,
|
|
629
637
|
y2: Math.round(cropper.y2 * ratio) - offsetY
|
|
630
638
|
};
|
|
631
|
-
if (!
|
|
639
|
+
if (!input.options?.containWithinAspectRatio) {
|
|
632
640
|
out.x1 = Math.max(out.x1, 0);
|
|
633
641
|
out.y1 = Math.max(out.y1, 0);
|
|
634
|
-
out.x2 = Math.min(out.x2,
|
|
635
|
-
out.y2 = Math.min(out.y2,
|
|
642
|
+
out.x2 = Math.min(out.x2, input.loadedImage.transformed.size.width);
|
|
643
|
+
out.y2 = Math.min(out.y2, input.loadedImage.transformed.size.height);
|
|
636
644
|
}
|
|
637
645
|
return out;
|
|
638
646
|
}
|
|
639
647
|
getResizeRatio(width, height, options) {
|
|
640
|
-
const ratioWidth = options.resizeToWidth / width;
|
|
641
|
-
const ratioHeight = options.resizeToHeight / height;
|
|
642
648
|
const ratios = new Array();
|
|
643
|
-
if (options.resizeToWidth > 0) {
|
|
644
|
-
ratios.push(
|
|
649
|
+
if (options?.resizeToWidth && options.resizeToWidth > 0) {
|
|
650
|
+
ratios.push(options.resizeToWidth / width);
|
|
645
651
|
}
|
|
646
|
-
if (options.resizeToHeight > 0) {
|
|
647
|
-
ratios.push(
|
|
652
|
+
if (options?.resizeToHeight && options.resizeToHeight > 0) {
|
|
653
|
+
ratios.push(options.resizeToHeight / height);
|
|
648
654
|
}
|
|
649
655
|
const result = ratios.length === 0 ? 1 : Math.min(...ratios);
|
|
650
|
-
if (result > 1 && !options
|
|
656
|
+
if (result > 1 && !options?.onlyScaleDown) {
|
|
651
657
|
return result;
|
|
652
658
|
}
|
|
653
659
|
return Math.min(result, 1);
|
|
654
660
|
}
|
|
655
661
|
getQuality(options) {
|
|
656
|
-
return Math.min(1, Math.max(0, options
|
|
662
|
+
return Math.min(1, Math.max(0, options?.imageQuality ?? 92 / 100));
|
|
657
663
|
}
|
|
658
664
|
}
|
|
659
665
|
|
|
@@ -1325,7 +1331,7 @@ class ImageCropperComponent {
|
|
|
1325
1331
|
}
|
|
1326
1332
|
cropToBlob() {
|
|
1327
1333
|
return new Promise(async (resolve, reject) => {
|
|
1328
|
-
const result = await this.cropService.crop(this.state, 'blob');
|
|
1334
|
+
const result = await this.cropService.crop(this.state.toCropInput(), 'blob');
|
|
1329
1335
|
if (result) {
|
|
1330
1336
|
this.imageCropped.emit(result);
|
|
1331
1337
|
resolve(result);
|
|
@@ -1336,7 +1342,7 @@ class ImageCropperComponent {
|
|
|
1336
1342
|
});
|
|
1337
1343
|
}
|
|
1338
1344
|
cropToBase64() {
|
|
1339
|
-
const result = this.cropService.crop(this.state, 'base64');
|
|
1345
|
+
const result = this.cropService.crop(this.state.toCropInput(), 'base64');
|
|
1340
1346
|
if (result) {
|
|
1341
1347
|
this.imageCropped.emit(result);
|
|
1342
1348
|
return result;
|