@libs-ui/components-image-editor 0.2.30-6.1

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.
@@ -0,0 +1,781 @@
1
+ /* eslint-disable @typescript-eslint/no-explicit-any */
2
+ import { NgTemplateOutlet } from '@angular/common';
3
+ import { ChangeDetectionStrategy, Component, DestroyRef, Inject, inject, input, model, Optional, output, signal, viewChild } from '@angular/core';
4
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
5
+ import { LibsUiComponentsButtonsButtonComponent } from '@libs-ui/components-buttons-button';
6
+ import { LibsUiComponentsInputsValidComponent } from '@libs-ui/components-inputs-valid';
7
+ import { LibsUiComponentsModalComponent } from '@libs-ui/components-modal';
8
+ import { LibsUiComponentsPopoverComponent } from '@libs-ui/components-popover';
9
+ import { LibsUiComponentsSpinnerComponent } from '@libs-ui/components-spinner';
10
+ import { LibsUiDynamicComponentService } from '@libs-ui/services-dynamic-component';
11
+ import { convertBase64ToBlob, get, isNil, LINK_IMAGE_ERROR_TOKEN_INJECT, set } from '@libs-ui/utils';
12
+ import { TranslateModule } from '@ngx-translate/core';
13
+ import { fromEvent } from 'rxjs';
14
+ import { mergeMap, takeUntil, tap } from 'rxjs/operators';
15
+ import { cropRationItems, getCropRectImage, getDataUrl, getStylesOfElement, getWidthHeightResizeCropFollow } from './defines/image-editor.define';
16
+ import { LibsUiComponentsImageEditorResizeComponent } from './resize/resize.component';
17
+ import * as i0 from "@angular/core";
18
+ import * as i1 from "@ngx-translate/core";
19
+ export class LibsUiComponentsImageEditorComponent {
20
+ linkImageError;
21
+ loading = signal(false);
22
+ loadingImage = signal(true);
23
+ changed = signal(false);
24
+ originWidth = signal(0);
25
+ originHeight = signal(0);
26
+ cropRatioItems = signal(cropRationItems());
27
+ cropRatioItemSelected = signal('');
28
+ cropSize = signal({ width: 20, height: 20 });
29
+ resizeData = signal({ ratio: 100, width: 600, height: 800 });
30
+ dragGridCrop = signal(false);
31
+ image = new Image();
32
+ canvas = document.createElement('canvas');
33
+ containerHeight = signal(800);
34
+ containerWidth = signal(1200);
35
+ imgContainerRect = signal({ top: 0, left: 0, width: 0, height: 0 });
36
+ rectClip = signal({ top: 0, left: 0, right: 0, bottom: 0, width: 0, height: 0 });
37
+ resizeState = signal('none');
38
+ moveState = signal('none');
39
+ startMouseDim = signal({ clientX: 0, clientY: 0, width: 0, height: 0, top: 0, left: 0, imageTop: 0, imageLeft: 0, imageWidth: 0, imageHeight: 0 });
40
+ dataUrlOrigin = signal('');
41
+ ratioValue = signal(0);
42
+ minHeight = 20;
43
+ minWidth = 20;
44
+ resizeComponentRef;
45
+ modeShowButton = input('save-file');
46
+ mimetype = input();
47
+ zIndex = input(1200);
48
+ imgSrc = model.required();
49
+ originUrl = input();
50
+ nameFile = input();
51
+ hasZoom = input(false);
52
+ aspectRatio = input();
53
+ requiredCropFollowRatio = input(false);
54
+ imageOrigin = viewChild.required('imageOrigin');
55
+ imageClip = viewChild.required('imageClip');
56
+ imageEditorContainer = viewChild.required('imageEditorContainer');
57
+ imageContainer = viewChild.required('imageContainer');
58
+ cropArea = viewChild.required('cropArea');
59
+ cropTL = viewChild.required('cropTL');
60
+ cropBL = viewChild.required('cropBL');
61
+ cropBR = viewChild.required('cropBR');
62
+ cropTR = viewChild.required('cropTR');
63
+ cropLineVL = viewChild.required('cropLineVL');
64
+ cropLineVR = viewChild.required('cropLineVR');
65
+ cropLineHT = viewChild.required('cropLineHT');
66
+ cropLineHB = viewChild.required('cropLineHB');
67
+ circleTL = viewChild.required('circleTL');
68
+ circleTR = viewChild.required('circleTR');
69
+ circleBL = viewChild.required('circleBL');
70
+ circleBR = viewChild.required('circleBR');
71
+ outClose = output();
72
+ outSaveFile = output();
73
+ outFunctionsControl = output();
74
+ destroyRef = inject(DestroyRef);
75
+ dynamicComponentService = inject(LibsUiDynamicComponentService);
76
+ constructor(linkImageError) {
77
+ this.linkImageError = linkImageError;
78
+ }
79
+ ngAfterViewInit() {
80
+ const aspectRatioFound = this.aspectRatio()?.key ? this.cropRatioItems().find(item => item.key === this.aspectRatio()?.key) : this.cropRatioItems()[0];
81
+ if (aspectRatioFound) {
82
+ this.dragGridCrop.set(true);
83
+ this.cropRatioItemSelected.set(aspectRatioFound?.key);
84
+ this.updateOriginImageSize();
85
+ this.changedImage(true);
86
+ }
87
+ this.updateModalSize();
88
+ this.initData();
89
+ this.initMouseEvent();
90
+ this.outFunctionsControl.emit({
91
+ cropImage: this.cropImage.bind(this),
92
+ setLoadingState: (loading) => this.loading.set(loading)
93
+ });
94
+ }
95
+ handlerImageLoaded() {
96
+ setTimeout(() => {
97
+ this.updateModalSize();
98
+ this.loadingImage.set(false);
99
+ }, 500);
100
+ }
101
+ handlerImageError(e) {
102
+ console.log('e', e);
103
+ e.target.src = this.linkImageError;
104
+ }
105
+ updateModalSize() {
106
+ const maxHeight = window.innerHeight - 100;
107
+ const maxWidth = window.innerWidth - 400;
108
+ this.containerHeight.set(maxHeight - 124);
109
+ this.containerWidth.set(maxWidth - 384);
110
+ this.updateOriginImageSize();
111
+ }
112
+ initData() {
113
+ this.imgContainerRect.set(this.imageContainer().nativeElement.getBoundingClientRect());
114
+ this.image.setAttribute('crossorigin', 'anonymous');
115
+ this.image.onload = () => {
116
+ this.imgContainerRect.set(this.imageContainer().nativeElement.getBoundingClientRect());
117
+ this.canvas.height = this.image.height;
118
+ this.canvas.width = this.image.width;
119
+ this.originHeight.set(this.image.height);
120
+ this.originWidth.set(this.image.width);
121
+ this.updateOriginImageSize();
122
+ this.canvas.getContext('2d')?.drawImage(this.image, 0, 0, this.canvas.width, this.canvas.height);
123
+ this.dataUrlOrigin.set(getDataUrl(this.canvas, this.mimetype(), this.image.src));
124
+ this.imgSrc.set(this.dataUrlOrigin());
125
+ };
126
+ this.image.src = this.imgSrc();
127
+ }
128
+ initMouseEvent() {
129
+ const mouseUp = this.initEvent({ nativeElement: document }, 'mouseup', { callStopPropagation: true, callPreventDefault: true }, undefined, () => this.handlerMouseup());
130
+ const mouseMove = this.initEvent({ nativeElement: document }, 'mousemove', { callStopPropagation: true, callPreventDefault: true }).pipe(takeUntil(mouseUp));
131
+ this.initEvent(this.imageEditorContainer(), 'wheel', { callStopPropagation: true }, undefined, (e) => this.handlerMousewheel(e)).subscribe();
132
+ this.initEvent({ nativeElement: window }, 'resize', {}, undefined, () => this.updateModalSize()).subscribe();
133
+ this.initEvent(this.imageContainer(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => this.handlerImageContainerMousedown(e)).subscribe((e) => this.handlerMousemove(e));
134
+ this.initEvent(this.cropTL(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'TL'); }).subscribe((e) => this.handlerMousemove(e));
135
+ this.initEvent(this.cropBL(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'BL'); }).subscribe((e) => this.handlerMousemove(e));
136
+ this.initEvent(this.cropBR(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'BR'); }).subscribe((e) => this.handlerMousemove(e));
137
+ this.initEvent(this.cropTR(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'TR'); }).subscribe((e) => this.handlerMousemove(e));
138
+ this.initEvent(this.cropLineVL(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'VL'); }).subscribe((e) => this.handlerMousemove(e));
139
+ this.initEvent(this.cropLineVR(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'VR'); }).subscribe((e) => this.handlerMousemove(e));
140
+ this.initEvent(this.cropLineHT(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'HT'); }).subscribe((e) => this.handlerMousemove(e));
141
+ this.initEvent(this.cropLineHB(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e) => { this.handlerCropResize(e, 'HB'); }).subscribe((e) => this.handlerMousemove(e));
142
+ this.initEvent(this.cropArea(), 'mousedown', {}, undefined, (e) => this.handlerMousedownCropArea(e)).subscribe((e) => this.handlerMousemove(e));
143
+ }
144
+ initEvent(element, eventName, flag, obsMerge, callback) {
145
+ const obs = fromEvent(element.nativeElement, eventName).pipe(tap((e) => {
146
+ if (flag.callStopPropagation && e?.stopPropagation) {
147
+ e.stopPropagation();
148
+ }
149
+ if (flag.callPreventDefault && e?.preventDefault) {
150
+ e.preventDefault();
151
+ }
152
+ if (callback) {
153
+ callback(e);
154
+ }
155
+ }), takeUntilDestroyed(this.destroyRef));
156
+ if (obsMerge) {
157
+ return obs.pipe(mergeMap(() => obsMerge), (takeUntilDestroyed(this.destroyRef)));
158
+ }
159
+ return obs;
160
+ }
161
+ ;
162
+ updateOriginImageSize() {
163
+ let width = this.originWidth();
164
+ let height = this.originHeight();
165
+ const ratio = width / height;
166
+ if (width >= height) {
167
+ width = Math.min(this.containerWidth(), width);
168
+ height = Math.min(width / ratio, this.containerHeight());
169
+ width = height * ratio;
170
+ }
171
+ else {
172
+ height = Math.min(this.containerHeight(), height);
173
+ width = height * ratio;
174
+ }
175
+ this.setImagePosition(width, height);
176
+ this.updateCropAreaPos();
177
+ this.updateRectClipPos();
178
+ }
179
+ setImagePosition(width, height) {
180
+ const left = this.containerWidth() / 2 - width / 2;
181
+ const top = this.containerHeight() / 2 - height / 2;
182
+ const styles = { width, height, top, left, display: 'block' };
183
+ this.setStylesElements(this.imageOrigin(), styles);
184
+ this.setStylesElements(this.imageClip(), styles);
185
+ this.setCirclePosition(styles);
186
+ }
187
+ setCirclePosition(styles) {
188
+ this.setStylesElements(this.circleTL(), { top: (styles.top || 0) - 5, left: (styles.left || 0) - 5 });
189
+ this.setStylesElements(this.circleTR(), { top: (styles.top || 0) - 5, left: (styles.left || 0) + (styles.width || 0) - 10 });
190
+ this.setStylesElements(this.circleBL(), { top: (styles.top || 0) + (styles.height || 0) - 10, left: (styles.left || 0) - 5 });
191
+ this.setStylesElements(this.circleBR(), { top: (styles.top || 0) + (styles.height || 0) - 10, left: (styles.left || 0) + (styles.width || 0) - 10 });
192
+ }
193
+ updateCropAreaPos() {
194
+ const [imgTop, imgLeft, imgWidth, imgHeight] = getStylesOfElement(this.imageClip().nativeElement, ['style.top', 'style.left', 'offsetWidth', 'offsetHeight']);
195
+ const rectImg = {
196
+ top: Math.max(0, imgTop),
197
+ left: Math.max(0, imgLeft),
198
+ width: imgWidth,
199
+ height: imgHeight
200
+ };
201
+ rectImg.width = Math.min(imgWidth, this.containerWidth() - imgLeft);
202
+ if (imgLeft < 0) {
203
+ rectImg.width = Math.min(imgWidth + imgLeft, this.containerWidth());
204
+ }
205
+ rectImg.height = Math.min(imgHeight, this.containerHeight() - imgTop);
206
+ if (imgTop < 0) {
207
+ rectImg.height = Math.min(imgTop + imgHeight, this.containerHeight());
208
+ }
209
+ this.ratioValue.set(this.getCropRatioValue());
210
+ if (!this.ratioValue()) {
211
+ this.setStylesElements(this.cropArea(), rectImg);
212
+ return;
213
+ }
214
+ let width = rectImg.width;
215
+ let height = width / this.ratioValue();
216
+ width = Math.min(rectImg.width, width);
217
+ height = width / this.ratioValue();
218
+ height = Math.min(rectImg.height, height);
219
+ width = height * this.ratioValue();
220
+ const top = rectImg.top + (rectImg.height - height) / 2;
221
+ const left = rectImg.left + (rectImg.width - width) / 2;
222
+ this.setStylesElements(this.cropArea(), { width, height, top, left });
223
+ }
224
+ getCropRatioValue() {
225
+ const cropRatio = this.cropRatioItems().find(item => item.key === this.cropRatioItemSelected());
226
+ if (!cropRatio?.value) {
227
+ return this.aspectRatio()?.value || (this.originWidth() / this.originHeight());
228
+ }
229
+ return cropRatio.value;
230
+ }
231
+ updateRectClipPos() {
232
+ const reactClip = this.getRectClipImage();
233
+ this.imageClip().nativeElement.style.clip = `rect(${reactClip.top}px, ${reactClip.right}px, ${reactClip.bottom}px, ${reactClip.left}px)`;
234
+ }
235
+ getRectClipImage() {
236
+ const imageRect = this.imageClip().nativeElement.getBoundingClientRect();
237
+ if (!imageRect.width) {
238
+ return this.rectClip();
239
+ }
240
+ const cropRect = this.cropArea().nativeElement.getBoundingClientRect();
241
+ const width = Math.round(cropRect.width);
242
+ const height = Math.round(cropRect.height);
243
+ const left = cropRect.left - imageRect.left;
244
+ const top = cropRect.top - imageRect.top;
245
+ const right = Math.round(left + width);
246
+ const bottom = Math.round(top + height);
247
+ const ratio = this.originWidth() / imageRect.width;
248
+ this.rectClip.set({ left, top, right, bottom, width, height });
249
+ this.cropSize.set({
250
+ height: top >= 0 ? (Math.round((bottom - top) * ratio)) : (Math.floor((bottom - top) * ratio)),
251
+ width: top >= 0 ? (Math.round(((right - left) * (this.ratioValue() ?? 1)) * ratio)) : (Math.floor(((right - left) * (this.ratioValue() ?? 1)) * ratio))
252
+ });
253
+ return this.rectClip();
254
+ }
255
+ handlerMousemove(e) {
256
+ const typeState = ['BL', 'TL', 'TR', 'BR', 'VL', 'VR', 'HT', 'HB'];
257
+ const methodName = `resizeCropFollow${this.resizeState()}Dir`; // resizeCropFollowBLDir, resizeCropFollowTLDir, resizeCropFollowTRDir, resizeCropFollowBRDir, resizeCropFollowVLDir, resizeCropFollowVRDir, resizeCropFollowHTDir, resizeCropFollowHBDir
258
+ if (typeState.includes(this.resizeState())) {
259
+ this[methodName](e);
260
+ }
261
+ switch (this.moveState()) {
262
+ case 'clip':
263
+ this.moveClipArea(e);
264
+ break;
265
+ case 'image':
266
+ this.moveImage(e);
267
+ break;
268
+ }
269
+ this.changedImage(true);
270
+ }
271
+ resizeCropFollowBLDir = (e) => {
272
+ const momentY = e.clientY - this.startMouseDim().clientY;
273
+ const momentX = this.ratioValue() ? -momentY * this.ratioValue() : e.clientX - this.startMouseDim().clientX;
274
+ const maxHeight = this.startMouseDim().imageTop + this.startMouseDim().imageHeight - this.startMouseDim().top;
275
+ const maxWidth = this.startMouseDim().left + this.startMouseDim().width - this.startMouseDim().imageLeft;
276
+ let { width, height, left } = this.getRectResizeCropFollow({ width: -1, height: 1, left: 1, top: 0 }, momentX, momentY);
277
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
278
+ left = this.startMouseDim().left + this.startMouseDim().width - width;
279
+ this.setStylesElements(this.cropArea(), { left, height, width }, true);
280
+ };
281
+ resizeCropFollowTLDir = (e) => {
282
+ const momentY = e.clientY - this.startMouseDim().clientY;
283
+ const momentX = this.ratioValue() ? this.ratioValue() * momentY : e.clientX - this.startMouseDim().clientX;
284
+ const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;
285
+ const maxWidth = this.startMouseDim().left + this.startMouseDim().width - this.startMouseDim().imageLeft;
286
+ let { width, height, top, left } = this.getRectResizeCropFollow({ width: -1, height: -1, left: 1, top: 1 }, momentX, momentY);
287
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
288
+ left = this.startMouseDim().left + this.startMouseDim().width - width;
289
+ top = this.startMouseDim().top + this.startMouseDim().height - height;
290
+ this.setStylesElements(this.cropArea(), { top, left, height, width }, true);
291
+ };
292
+ resizeCropFollowTRDir = (e) => {
293
+ const momentY = e.clientY - this.startMouseDim().clientY;
294
+ const momentX = this.ratioValue() ? -momentY * this.ratioValue() : e.clientX - this.startMouseDim().clientX;
295
+ const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;
296
+ const maxWidth = this.startMouseDim().imageLeft + this.startMouseDim().imageWidth - this.startMouseDim().left;
297
+ let { width, height, top } = this.getRectResizeCropFollow({ width: 1, height: -1, left: 0, top: 1 }, momentX, momentY);
298
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
299
+ top = this.startMouseDim().top + this.startMouseDim().height - height;
300
+ this.setStylesElements(this.cropArea(), { top, height, width }, true);
301
+ };
302
+ resizeCropFollowBRDir(e) {
303
+ const momentY = e.clientY - this.startMouseDim().clientY;
304
+ const momentX = this.ratioValue() ? momentY * this.ratioValue() : e.clientX - this.startMouseDim().clientX;
305
+ const maxHeight = this.startMouseDim().imageTop + this.startMouseDim().imageHeight - this.startMouseDim().top;
306
+ const maxWidth = this.startMouseDim().imageLeft + this.startMouseDim().imageWidth - this.startMouseDim().left;
307
+ let { width, height } = this.getRectResizeCropFollow({ width: 1, height: 1, left: 0, top: 0 }, momentX, momentY);
308
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
309
+ this.setStylesElements(this.cropArea(), { height, width }, true);
310
+ }
311
+ resizeCropFollowVLDir = (e) => {
312
+ const momentX = e.clientX - this.startMouseDim().clientX;
313
+ const momentY = this.ratioValue() ? momentX / this.ratioValue() : 0;
314
+ const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;
315
+ const maxWidth = this.startMouseDim().left + this.startMouseDim().width - this.startMouseDim().imageLeft;
316
+ let { width, height, top, left } = this.getRectResizeCropFollow({ width: -1, height: -1, left: 1, top: 1 }, momentX, momentY);
317
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
318
+ left = this.startMouseDim().left + this.startMouseDim().width - width;
319
+ top = this.startMouseDim().top + this.startMouseDim().height - height;
320
+ this.setStylesElements(this.cropArea(), { top, left, height, width }, true);
321
+ };
322
+ resizeCropFollowVRDir = (e) => {
323
+ const momentX = e.clientX - this.startMouseDim().clientX;
324
+ const momentY = this.ratioValue() ? -momentX / this.ratioValue() : 0;
325
+ const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;
326
+ const maxWidth = this.startMouseDim().imageLeft + this.startMouseDim().imageWidth - this.startMouseDim().left;
327
+ let { width, height, top } = this.getRectResizeCropFollow({ width: 1, height: -1, left: 0, top: 1 }, momentX, momentY);
328
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
329
+ top = this.startMouseDim().top + this.startMouseDim().height - height;
330
+ this.setStylesElements(this.cropArea(), { top, height, width }, true);
331
+ };
332
+ resizeCropFollowHTDir = (e) => {
333
+ const momentY = e.clientY - this.startMouseDim().clientY;
334
+ const momentX = this.ratioValue() ? momentY * this.ratioValue() : 0;
335
+ const maxWidth = this.startMouseDim().width + this.startMouseDim().left - this.startMouseDim().imageLeft;
336
+ const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;
337
+ let { width, height, top, left } = this.getRectResizeCropFollow({ width: -1, height: -1, left: 1, top: 1 }, momentX, momentY);
338
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
339
+ top = this.startMouseDim().top + this.startMouseDim().height - height;
340
+ left = this.startMouseDim().left + this.startMouseDim().width - width;
341
+ this.setStylesElements(this.cropArea(), { top, left, height, width }, true);
342
+ };
343
+ resizeCropFollowHBDir = (e) => {
344
+ const momentY = e.clientY - this.startMouseDim().clientY;
345
+ const momentX = this.ratioValue() ? momentY * this.ratioValue() : 0;
346
+ const maxHeight = this.startMouseDim().imageTop + this.startMouseDim().imageHeight - this.startMouseDim().top;
347
+ const maxWidth = this.startMouseDim().width + this.startMouseDim().left - this.startMouseDim().imageLeft;
348
+ let { width, height, left } = this.getRectResizeCropFollow({ width: 1, height: 1, left: -1, top: 0 }, momentX, momentY);
349
+ [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);
350
+ left = this.startMouseDim().left + this.startMouseDim().width - width;
351
+ this.setStylesElements(this.cropArea(), { height, left, width }, true);
352
+ };
353
+ getRectResizeCropFollow(ratio, momentX, momentY) {
354
+ const width = this.startMouseDim().width + momentX * ratio.width;
355
+ const height = this.startMouseDim().height + momentY * ratio.height;
356
+ const top = this.startMouseDim().top + momentY * ratio.top;
357
+ const left = this.startMouseDim().left + momentX * ratio.left;
358
+ return { width, height, left, top };
359
+ }
360
+ handlerMouseup() {
361
+ this.imageContainer().nativeElement.classList.remove('cursor-grabbing');
362
+ this.cropArea().nativeElement.classList.remove('cursor-grabbing');
363
+ this.resizeState.set('none');
364
+ this.moveState.set('none');
365
+ }
366
+ handlerMousewheel(event) {
367
+ if (!this.hasZoom()) {
368
+ return;
369
+ }
370
+ if (event.deltaY < 0) {
371
+ this.zoomImage(1 / 1.1);
372
+ return;
373
+ }
374
+ this.zoomImage(1.1);
375
+ }
376
+ handlerImageContainerMousedown(e) {
377
+ if (this.resizeState() === 'none' && this.moveState() === 'none') {
378
+ this.moveState.set('image');
379
+ this.handlerMousedown(e);
380
+ }
381
+ return false;
382
+ }
383
+ handlerCropResize(e, resizeType) {
384
+ this.resizeState.set(resizeType);
385
+ this.dragGridCrop.set(true);
386
+ this.handlerMousedown(e);
387
+ }
388
+ handlerMousedownCropArea(e) {
389
+ this.moveState.set('clip');
390
+ this.handlerMousedown(e);
391
+ }
392
+ handlerMousedown(e) {
393
+ const [width, height, top, left] = getStylesOfElement(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);
394
+ const [imageWidth, imageHeight, imageTop, imageLeft] = getStylesOfElement(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);
395
+ switch (this.moveState()) {
396
+ case 'image':
397
+ this.imageContainer().nativeElement.classList.add('cursor-grabbing');
398
+ break;
399
+ case 'clip':
400
+ if (width === imageWidth && height === imageHeight) {
401
+ return;
402
+ }
403
+ this.cropArea().nativeElement.classList.add('cursor-grabbing');
404
+ break;
405
+ default:
406
+ break;
407
+ }
408
+ this.startMouseDim.set({
409
+ clientX: e.clientX,
410
+ clientY: e.clientY,
411
+ width: width,
412
+ height: height,
413
+ top: top,
414
+ left: left,
415
+ imageTop: imageTop,
416
+ imageLeft: imageLeft,
417
+ imageWidth: imageWidth,
418
+ imageHeight: imageHeight
419
+ });
420
+ this.ratioValue.set(this.getCropRatioValue());
421
+ }
422
+ moveImage(e) {
423
+ const momentY = e.clientY - this.startMouseDim().clientY;
424
+ const momentX = e.clientX - this.startMouseDim().clientX;
425
+ const imgTop = this.startMouseDim().imageTop + momentY;
426
+ const imgLeft = this.startMouseDim().imageLeft + momentX;
427
+ const top = this.startMouseDim().top + momentY;
428
+ const left = this.startMouseDim().left + momentX;
429
+ this.setStylesElements(this.imageOrigin(), { top: imgTop, left: imgLeft });
430
+ this.setStylesElements(this.imageClip(), { top: imgTop, left: imgLeft });
431
+ this.setStylesElements(this.cropArea(), { top, left });
432
+ this.setCirclePosition({ top: imgTop, left: imgLeft, width: this.startMouseDim().imageWidth, height: this.startMouseDim().imageHeight });
433
+ }
434
+ moveClipArea(e) {
435
+ const [imgWidth, imgHeight, imgTop, imgLeft] = getStylesOfElement(this.imageClip().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);
436
+ const [cropWidth, cropHeight] = getStylesOfElement(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight']);
437
+ const momentY = e.clientY - this.startMouseDim().clientY;
438
+ const momentX = e.clientX - this.startMouseDim().clientX;
439
+ let top = Math.max(imgTop, this.startMouseDim().top + momentY);
440
+ let left = Math.max(imgLeft, this.startMouseDim().left + momentX);
441
+ top = Math.min(top, imgTop + imgHeight - cropHeight);
442
+ left = Math.min(left, imgLeft + imgWidth - cropWidth);
443
+ this.setStylesElements(this.cropArea(), { top, left }, true);
444
+ }
445
+ zoomImage(scale) {
446
+ const [currWidth, currHeight, currTop, currLeft] = getStylesOfElement(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);
447
+ const width = Math.max(currWidth * scale, 50);
448
+ const height = this.imageOrigin().nativeElement.offsetHeight * width / this.imageOrigin().nativeElement.offsetWidth;
449
+ // Calculate the center point of the current view
450
+ const centerX = currLeft + currWidth / 2;
451
+ const centerY = currTop + currHeight / 2;
452
+ // Calculate new position to maintain center point
453
+ const top = centerY - height / 2;
454
+ const left = centerX - width / 2;
455
+ // Get current crop area dimensions and position
456
+ let [cropWidth, cropHeight] = getStylesOfElement(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight']);
457
+ const [cropTop, cropLeft] = getStylesOfElement(this.cropArea().nativeElement, ['style.top', 'style.left']);
458
+ // Scale crop area dimensions proportionally
459
+ cropWidth = cropWidth * scale;
460
+ cropHeight = cropHeight * scale;
461
+ // Calculate new crop area position relative to the image
462
+ const cropTopNew = top + (cropTop - currTop) * scale;
463
+ const cropLeftNew = left + (cropLeft - currLeft) * scale;
464
+ // Ensure crop area stays within bounds
465
+ const maxCropWidth = width;
466
+ const maxCropHeight = height;
467
+ const minWidth = 20;
468
+ const minHeight = 20;
469
+ cropWidth = Math.min(cropWidth, maxCropWidth);
470
+ cropHeight = Math.min(cropHeight, maxCropHeight);
471
+ cropWidth = Math.max(cropWidth, minWidth);
472
+ cropHeight = Math.max(cropHeight, minHeight);
473
+ this.ratioValue.set(this.getCropRatioValue());
474
+ if (this.ratioValue()) {
475
+ cropHeight = cropWidth / this.ratioValue();
476
+ cropHeight = Math.min(cropHeight, maxCropHeight);
477
+ cropHeight = Math.max(cropHeight, minHeight);
478
+ cropWidth = cropHeight * this.ratioValue();
479
+ }
480
+ // Update clipping rectangle
481
+ this.rectClip.update(rect => {
482
+ const rectTop = cropTopNew - top;
483
+ const rectLeft = cropLeftNew - left;
484
+ return {
485
+ ...rect,
486
+ top: rectTop,
487
+ left: rectLeft,
488
+ right: rectLeft + cropWidth,
489
+ bottom: rectTop + cropHeight
490
+ };
491
+ });
492
+ const clip = `rect(${this.rectClip().top}px, ${this.rectClip().right}px, ${this.rectClip().bottom}px, ${this.rectClip().left}px)`;
493
+ // Update all elements with new positions and dimensions
494
+ this.setStylesElements(this.cropArea(), { top: cropTopNew, left: cropLeftNew, width: cropWidth, height: cropHeight });
495
+ this.setStylesElements(this.imageOrigin(), { top, left, width, height });
496
+ this.setCirclePosition({ top, left, width, height });
497
+ this.setStylesElements(this.imageClip(), { clip, top, left, width, height });
498
+ this.getRectClipImage();
499
+ }
500
+ handlerRotateImage(event) {
501
+ event.stopPropagation();
502
+ this.image.src = this.imgSrc();
503
+ this.image.onload = () => {
504
+ const canvas = document.createElement('canvas');
505
+ const ctx = canvas.getContext("2d");
506
+ const [width, height] = getStylesOfElement(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight']);
507
+ const originWidth = this.image.width;
508
+ const originHeight = this.image.height;
509
+ let top = parseInt(this.imageOrigin().nativeElement.style.top);
510
+ let left = parseInt(this.imageOrigin().nativeElement.style.left);
511
+ let newHeight = width;
512
+ let newWidth = height;
513
+ const ratio = newWidth / newHeight;
514
+ canvas.height = originWidth;
515
+ canvas.width = originHeight;
516
+ ctx?.translate(originHeight / 2, originWidth / 2);
517
+ ctx?.rotate(Math.PI / 2);
518
+ ctx?.drawImage(this.image, -originWidth / 2, -originHeight / 2);
519
+ const src = getDataUrl(canvas, this.mimetype(), this.image.src);
520
+ this.imgSrc.set(src);
521
+ newWidth = Math.max(newWidth, originHeight);
522
+ newWidth = Math.min(newWidth, this.containerWidth());
523
+ newHeight = newWidth / ratio;
524
+ newHeight = Math.max(newHeight, originWidth);
525
+ newHeight = Math.min(newHeight, this.containerHeight());
526
+ newWidth = newHeight * ratio;
527
+ top = top - (newHeight / 2 - height / 2);
528
+ left = left - (newWidth / 2 - width / 2);
529
+ this.setStylesElements(this.imageOrigin(), { src, top, left, width: newWidth, height: newHeight });
530
+ this.setCirclePosition({ top, left, width: newWidth, height: newHeight });
531
+ this.setStylesElements(this.imageClip(), { src, top, left, width: newWidth, height: newHeight });
532
+ this.getOriginalImageSize(() => {
533
+ this.cropSize.set({ width: this.originWidth(), height: this.originHeight() });
534
+ this.updateOriginImageSize();
535
+ });
536
+ this.changedImage(true);
537
+ };
538
+ }
539
+ getOriginalImageSize(callback) {
540
+ this.image.src = this.imgSrc();
541
+ this.image.onload = () => {
542
+ this.originWidth.set(this.image.width);
543
+ this.originHeight.set(this.image.height);
544
+ if (callback) {
545
+ return callback();
546
+ }
547
+ };
548
+ }
549
+ changedImage(status) {
550
+ this.changed.set(status);
551
+ }
552
+ handlerFlipImage(event, mode) {
553
+ event.stopPropagation();
554
+ this.image.src = this.imgSrc();
555
+ this.image.onload = () => {
556
+ const canvas = document.createElement('canvas');
557
+ const ctx = canvas.getContext("2d");
558
+ const width = this.image.width;
559
+ const height = this.image.height;
560
+ canvas.height = height;
561
+ canvas.width = width;
562
+ if (ctx) {
563
+ ctx.drawImage(this.image, 0, 0, width, height);
564
+ if (mode === 'horizontal') {
565
+ ctx.translate(width, 0);
566
+ ctx.scale(-1, 1);
567
+ }
568
+ if (mode === 'vertical') {
569
+ ctx.translate(0, height);
570
+ ctx.scale(1, -1);
571
+ }
572
+ ctx.clearRect(0, 0, width, height);
573
+ ctx.drawImage(this.image, 0, 0);
574
+ }
575
+ const dataUrl = getDataUrl(canvas, this.mimetype(), this.image.src);
576
+ this.imgSrc.set(dataUrl);
577
+ this.imageOrigin().nativeElement.src = dataUrl;
578
+ this.imageClip().nativeElement.src = dataUrl;
579
+ this.updateCropAreaPos();
580
+ this.updateRectClipPos();
581
+ this.changedImage(true);
582
+ };
583
+ }
584
+ handlerRestoreImage(event) {
585
+ event.stopPropagation();
586
+ this.imgSrc.set(this.dataUrlOrigin());
587
+ this.dragGridCrop.set(false);
588
+ this.getOriginalImageSize(() => {
589
+ this.updateOriginImageSize();
590
+ this.changedImage(false);
591
+ this.cropRatioItemSelected.set('');
592
+ this.updateCropAreaPos();
593
+ this.updateRectClipPos();
594
+ });
595
+ }
596
+ cropImage() {
597
+ return new Promise((resolve) => {
598
+ this.image.src = this.imgSrc();
599
+ this.image.onload = () => {
600
+ const canvas = document.createElement('canvas');
601
+ const originWidth = this.image.width;
602
+ const width = parseInt(this.imageClip().nativeElement.offsetWidth);
603
+ const height = parseInt(this.imageClip().nativeElement.offsetHeight);
604
+ const scale = originWidth / width;
605
+ const rectClip = this.getRectClipImage();
606
+ const cropRect = getCropRectImage(rectClip, width, height, scale);
607
+ canvas.height = cropRect.height;
608
+ canvas.width = cropRect.width;
609
+ this.originHeight.set(canvas.height);
610
+ this.originWidth.set(canvas.width);
611
+ const ctx = canvas.getContext("2d");
612
+ if (ctx) {
613
+ ctx.drawImage(this.image, cropRect.left, cropRect.top, cropRect.width, cropRect.height, 0, 0, cropRect.width, cropRect.height);
614
+ }
615
+ const dataUrl = getDataUrl(canvas, this.mimetype(), this.image.src);
616
+ this.changedImage(true);
617
+ resolve(dataUrl);
618
+ };
619
+ });
620
+ }
621
+ async handlerSaveFile(event, modeSave) {
622
+ event.stopPropagation();
623
+ const dataUrl = await this.cropImage();
624
+ const file = convertBase64ToBlob(dataUrl);
625
+ this.outSaveFile.emit({ file: file, url: dataUrl, mode: modeSave });
626
+ }
627
+ async handlerClose(event) {
628
+ event.stopPropagation();
629
+ this.outClose.emit({ isClickButtonClose: true });
630
+ }
631
+ handlerSelectCropRatioItem(event, key) {
632
+ event.stopPropagation();
633
+ this.cropRatioItemSelected.set(key);
634
+ this.changed.set(true);
635
+ this.dragGridCrop.set(true);
636
+ this.updateOriginImageSize();
637
+ }
638
+ handlerResize(event) {
639
+ event.stopPropagation();
640
+ if (this.resizeComponentRef) {
641
+ return;
642
+ }
643
+ this.resizeComponentRef = this.dynamicComponentService.resolveComponentFactory(LibsUiComponentsImageEditorResizeComponent);
644
+ const instance = this.resizeComponentRef.instance;
645
+ const subs = instance.outClose.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(() => {
646
+ this.dynamicComponentService.remove(this.resizeComponentRef);
647
+ this.resizeComponentRef = undefined;
648
+ subs.unsubscribe();
649
+ });
650
+ instance.src = this.imgSrc();
651
+ instance.mimetype = this.mimetype();
652
+ instance.zIndex = this.zIndex();
653
+ subs.add(instance.outSave.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(event => {
654
+ this.resizeData.set(event);
655
+ this.saveResize();
656
+ }));
657
+ instance.resizeData = { ratio: 100, width: this.originWidth(), height: this.originHeight() };
658
+ instance.originWidth = this.originWidth();
659
+ instance.originHeight = this.originHeight();
660
+ this.dynamicComponentService.addToBody(this.resizeComponentRef);
661
+ }
662
+ saveResize() {
663
+ this.image.src = this.imgSrc();
664
+ this.image.onload = () => {
665
+ const canvas = document.createElement('canvas');
666
+ canvas.width = this.resizeData().width;
667
+ canvas.height = this.resizeData().height;
668
+ const ctx = canvas.getContext('2d');
669
+ if (ctx) {
670
+ ctx.drawImage(this.image, 0, 0, canvas.width, canvas.height);
671
+ }
672
+ const src = getDataUrl(canvas, this.mimetype(), this.image.src);
673
+ const [currWidth, currHeight, currTop, currLeft] = getStylesOfElement(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);
674
+ const ratio = currWidth / currHeight;
675
+ let newWidth = Math.min(this.resizeData().width, this.containerWidth());
676
+ let newHeight = newWidth / ratio;
677
+ newHeight = Math.min(this.resizeData().height, this.containerHeight());
678
+ newWidth = newHeight * ratio;
679
+ Math.min(this.resizeData().height, this.containerHeight());
680
+ const top = currTop + (currHeight - newHeight) / 2;
681
+ const left = currLeft + (currWidth - newWidth) / 2;
682
+ const movementY = currTop - top;
683
+ const movementX = currLeft - left;
684
+ this.rectClip.update(rect => {
685
+ const rectTop = rect.top + movementY;
686
+ const rectLeft = rect.left + movementX;
687
+ return {
688
+ ...rect,
689
+ top: rectTop,
690
+ left: rectLeft,
691
+ right: rectLeft + newWidth,
692
+ bottom: rectTop + newHeight
693
+ };
694
+ });
695
+ const clip = `rect(${this.rectClip().top}px, ${this.rectClip().right}px, ${this.rectClip().bottom}px, ${this.rectClip().left}px)`;
696
+ this.setStylesElements(this.imageOrigin(), { src, top, left, width: newWidth, height: newHeight });
697
+ this.setCirclePosition({ top, left, width: newWidth, height: newHeight });
698
+ this.setStylesElements(this.imageClip(), { src, clip, top, left, width: newWidth, height: newHeight });
699
+ this.imgSrc.set(src);
700
+ this.originWidth.set(this.resizeData().width);
701
+ this.originHeight.set(this.resizeData().height);
702
+ this.updateCropAreaPos();
703
+ this.updateRectClipPos();
704
+ this.changedImage(true);
705
+ };
706
+ }
707
+ handlerCropWidth() {
708
+ this.processCropWidthOrHeight('width');
709
+ }
710
+ handlerCropHeight() {
711
+ this.processCropWidthOrHeight('height');
712
+ }
713
+ processCropWidthOrHeight(cropBy) {
714
+ this.ratioValue.set(this.getCropRatioValue());
715
+ const [currImgWidth, currImgHeight, currImgTop, currImgLeft] = getStylesOfElement(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);
716
+ let [cropWidth, cropHeight] = getStylesOfElement(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight']);
717
+ const [cropTop, cropLeft] = getStylesOfElement(this.cropArea().nativeElement, ['style.top', 'style.left']);
718
+ const maxCropWidth = currImgLeft + currImgWidth - cropLeft;
719
+ const maxCropHeight = currImgHeight + currImgTop - cropTop;
720
+ const ratio = this.originWidth() / currImgWidth;
721
+ let cropValue = get(this.cropSize, cropBy) / ratio;
722
+ cropValue = Math.min(cropValue, cropBy === 'width' ? maxCropWidth : maxCropHeight);
723
+ cropValue = Math.max(1, cropValue);
724
+ if (this.ratioValue()) {
725
+ if (cropBy === 'width') {
726
+ cropHeight = cropValue / this.ratioValue();
727
+ cropHeight = Math.min(cropHeight, maxCropHeight);
728
+ cropValue = cropHeight * this.ratioValue();
729
+ this.cropSize.update(cropSize => ({ ...cropSize, height: Math.round(cropHeight * ratio) }));
730
+ this.cropArea().nativeElement.style.height = `${cropHeight}px`;
731
+ }
732
+ if (cropBy === 'height') {
733
+ cropWidth = cropValue * this.ratioValue();
734
+ cropWidth = Math.min(cropWidth, maxCropWidth);
735
+ cropValue = cropWidth / this.ratioValue();
736
+ this.cropSize.update(cropSize => ({ ...cropSize, width: Math.round(cropWidth * ratio) }));
737
+ this.cropArea().nativeElement.style.width = `${cropWidth}px`;
738
+ }
739
+ }
740
+ this.cropSize.update(cropSize => ({ ...cropSize, [cropBy]: Math.round(cropValue * ratio) }));
741
+ this.cropArea().nativeElement.style[cropBy] = `${cropValue}px`;
742
+ const imageRect = this.imageClip().nativeElement.getBoundingClientRect();
743
+ const cropRect = this.cropArea().nativeElement.getBoundingClientRect();
744
+ const { width, height } = cropRect;
745
+ const left = cropRect.left - imageRect.left;
746
+ const top = cropRect.top - imageRect.top;
747
+ const right = left + width;
748
+ const bottom = top + height;
749
+ this.imageClip().nativeElement.style.clip = `rect(${top}px, ${right}px, ${bottom}px, ${left}px)`;
750
+ }
751
+ setStylesElements(element, styles, isUpdateRectClipPos) {
752
+ Object.keys(styles).forEach(key => {
753
+ const value = get(styles, key);
754
+ if (isNil(value)) {
755
+ return;
756
+ }
757
+ set(element.nativeElement.style, key, typeof value === 'number' ? `${value}px` : value);
758
+ });
759
+ if (isUpdateRectClipPos) {
760
+ this.updateRectClipPos();
761
+ }
762
+ }
763
+ ngOnDestroy() {
764
+ this.dynamicComponentService.remove(this.resizeComponentRef);
765
+ }
766
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LibsUiComponentsImageEditorComponent, deps: [{ token: LINK_IMAGE_ERROR_TOKEN_INJECT, optional: true }], target: i0.ɵɵFactoryTarget.Component });
767
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: LibsUiComponentsImageEditorComponent, isStandalone: true, selector: "libs_ui-components-image_editor", inputs: { modeShowButton: { classPropertyName: "modeShowButton", publicName: "modeShowButton", isSignal: true, isRequired: false, transformFunction: null }, mimetype: { classPropertyName: "mimetype", publicName: "mimetype", isSignal: true, isRequired: false, transformFunction: null }, zIndex: { classPropertyName: "zIndex", publicName: "zIndex", isSignal: true, isRequired: false, transformFunction: null }, imgSrc: { classPropertyName: "imgSrc", publicName: "imgSrc", isSignal: true, isRequired: true, transformFunction: null }, originUrl: { classPropertyName: "originUrl", publicName: "originUrl", isSignal: true, isRequired: false, transformFunction: null }, nameFile: { classPropertyName: "nameFile", publicName: "nameFile", isSignal: true, isRequired: false, transformFunction: null }, hasZoom: { classPropertyName: "hasZoom", publicName: "hasZoom", isSignal: true, isRequired: false, transformFunction: null }, aspectRatio: { classPropertyName: "aspectRatio", publicName: "aspectRatio", isSignal: true, isRequired: false, transformFunction: null }, requiredCropFollowRatio: { classPropertyName: "requiredCropFollowRatio", publicName: "requiredCropFollowRatio", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { imgSrc: "imgSrcChange", outClose: "outClose", outSaveFile: "outSaveFile", outFunctionsControl: "outFunctionsControl" }, viewQueries: [{ propertyName: "imageOrigin", first: true, predicate: ["imageOrigin"], descendants: true, isSignal: true }, { propertyName: "imageClip", first: true, predicate: ["imageClip"], descendants: true, isSignal: true }, { propertyName: "imageEditorContainer", first: true, predicate: ["imageEditorContainer"], descendants: true, isSignal: true }, { propertyName: "imageContainer", first: true, predicate: ["imageContainer"], descendants: true, isSignal: true }, { propertyName: "cropArea", first: true, predicate: ["cropArea"], descendants: true, isSignal: true }, { propertyName: "cropTL", first: true, predicate: ["cropTL"], descendants: true, isSignal: true }, { propertyName: "cropBL", first: true, predicate: ["cropBL"], descendants: true, isSignal: true }, { propertyName: "cropBR", first: true, predicate: ["cropBR"], descendants: true, isSignal: true }, { propertyName: "cropTR", first: true, predicate: ["cropTR"], descendants: true, isSignal: true }, { propertyName: "cropLineVL", first: true, predicate: ["cropLineVL"], descendants: true, isSignal: true }, { propertyName: "cropLineVR", first: true, predicate: ["cropLineVR"], descendants: true, isSignal: true }, { propertyName: "cropLineHT", first: true, predicate: ["cropLineHT"], descendants: true, isSignal: true }, { propertyName: "cropLineHB", first: true, predicate: ["cropLineHB"], descendants: true, isSignal: true }, { propertyName: "circleTL", first: true, predicate: ["circleTL"], descendants: true, isSignal: true }, { propertyName: "circleTR", first: true, predicate: ["circleTR"], descendants: true, isSignal: true }, { propertyName: "circleBL", first: true, predicate: ["circleBL"], descendants: true, isSignal: true }, { propertyName: "circleBR", first: true, predicate: ["circleBR"], descendants: true, isSignal: true }], ngImport: i0, template: "<libs_ui-components-modal [width]=\"'calc(100vw - 400px)'\"\n [height]=\"'calc(100vh - 100px)'\"\n [minWidth]=\"'1200px'\"\n [headerConfig]=\"{ignoreHeaderTheme: true, removeButtonClose: true}\"\n [bodyConfig]=\"{classInclude: '!p-0', scrollOverlayOptions: {scrollX: 'hidden', scrollY: 'hidden'}}\"\n [footerConfig]=\"{hidden: true}\"\n [zIndex]=\"zIndex()\"\n [buttonsFooter]=\"[]\"\n [mode]=\"'center'\">\n <div class=\"libs-ui-modal-header-custom w-full flex items-center justify-between\">\n <div class=\"w-full relative mx-[24px] h-[16px]\">\n <div class=\"flex w-full libs-ui-font-h4s absolute\">\n <libs_ui-components-popover [type]=\"'text'\"\n [config]=\"{zIndex: zIndex()}\">\n {{ 'i18n_edit_file_name' | translate:{file_name: nameFile()} }}\n </libs_ui-components-popover>\n </div>\n </div>\n <div class=\"flex items-center\">\n @if (modeShowButton() === 'save-file') {\n <libs_ui-components-buttons-button [type]=\"'button-third'\"\n [label]=\"'i18n_cancel'\"\n [disable]=\"loading() || loadingImage()\"\n [classInclude]=\"'py-[6px] px-[12px] mr-[16px]'\"\n (outClick)=\"handlerClose($event)\" />\n\n <libs_ui-components-buttons-button class=\"mr-[24px]\"\n [label]=\"'i18n_save'\"\n [disable]=\"loading() || loadingImage()\"\n [classInclude]=\"'py-[6px] px-[12px]'\"\n (outClick)=\"handlerSaveFile($event,'save-file')\" />\n }\n @else {\n <libs_ui-components-buttons-button [label]=\"'i18n_save_to_new_file'\"\n [type]=\"'button-outline'\"\n [classInclude]=\"'mx-[16px] py-[6px] px-[12px]'\"\n [disable]=\"loading() || loadingImage() || !changed()\"\n (outClick)=\"handlerSaveFile($event,'save-api-as-new-file')\" />\n <libs_ui-components-buttons-button [type]=\"'button-secondary'\"\n [label]=\"'i18n_save'\"\n [classInclude]=\"'py-[6px] px-[12px]'\"\n [disable]=\"loading() || loadingImage() || !changed()\"\n (outClick)=\"handlerSaveFile($event,'save-api')\" />\n <div class=\"h-[12px] libs-ui-border-left-general mx-[16px]\"></div>\n <libs_ui-components-buttons-button class=\"mr-[24px]\"\n [classInclude]=\"'p-[6px]'\"\n [type]=\"'button-third-hover-danger'\"\n [iconOnlyType]=\"true\"\n [disable]=\"loading()\"\n [popover]=\"{config: {content: 'i18n_exit_edit_mode', zIndex: zIndex()}}\"\n [classIconLeft]=\"'libs-ui-icon-close'\"\n (outClick)=\"handlerClose($event)\" />\n }\n </div>\n </div>\n <div class=\"libs-ui-modal-body-custom h-full w-full flex\">\n <div #imageEditorContainer\n class=\"relative w-full h-full bg-[#e6e7ea] border-radius-bottom-left-8px p-[32px] libs-ui-input-image-editor-container-left\">\n @if (loadingImage()) {\n <div class=\"absolute h-full w-full bg-[#e6e7ea] top-0 left-0 z-[5]\">\n <libs_ui-components-spinner [size]=\"'medium'\" />\n </div>\n }\n <div class=\"absolute h-[32px] w-[calc(100%-64px)] bg-[#e6e7eab8] top-0 left-[32px] z-[1]\">\n </div>\n <div class=\"absolute w-[32px] h-full bg-[#e6e7eab8] left-0 top-0 z-[1] rounded-bl-[8px]\">\n </div>\n <div class=\"absolute w-[32px] h-full bg-[#e6e7eab8] right-0 top-0 z-[1]\">\n </div>\n <div class=\"absolute h-[32px] w-[calc(100%-64px)] bg-[#e6e7eab8] bottom-0 left-[32px] z-[1]\">\n </div>\n <div #imageContainer\n class=\"relative w-full h-full rounded-bl-[8px] min-h-0\">\n <div #circleTL\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <div #circleTR\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <div #circleBL\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <div #circleBR\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <img #imageOrigin\n class=\"absolute border-[4px] border-[#8e61ee] rounded-[4px] top-[-1000px] left-[-1000px]\"\n loading=\"lazy\"\n [src]=\"imgSrc()\"\n (load)=\"handlerImageLoaded()\"\n (error)=\"handlerImageError($event)\" />\n <img #imageClip\n class=\"absolute z-[1] top-[-1000px] left-[-1000px]\"\n loading=\"lazy\"\n [src]=\"imgSrc()\"\n (error)=\"handlerImageError($event)\" />\n <div class=\"absolute top-0 left-0 right-0 bottom-0 bg-[#e6e7eab8]\">\n </div>\n <div #cropArea\n class=\"absolute z-[2] border-[4px] border-[#8e61ee] rounded-[4px]\">\n <div class=\"relative w-full h-full\">\n <div #cropTL\n class=\"absolute z-[2] left-[-18px] top-[-17px] cursor-nwse-resize\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropBL\n class=\"absolute z-[2] left-[-17px] bottom-[-18px] cursor-nesw-resize rotate-[270deg]\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropTR\n class=\"absolute z-[2] right-[-17px] top-[-17px] cursor-nesw-resize rotate-[90deg]\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropBR\n class=\"absolute z-[2] right-[-18px] bottom-[-17px] cursor-nwse-resize rotate-[180deg]\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropLineVL\n class=\"absolute z-[1] w-[12px] left-[-6px] top-0 bottom-0 cursor-ew-resize\">\n </div>\n <div #cropLineVR\n class=\"absolute z-[1] w-[12px] right-[-6px] top-0 bottom-0 cursor-ew-resize\">\n </div>\n <div #cropLineHT\n class=\"absolute z-[1] h-[12px] left-0 top-[-6px] right-0 cursor-ns-resize\">\n </div>\n <div #cropLineHB\n class=\"absolute z-[1] h-[12px] left-0 right-0 bottom-[-6px] cursor-ns-resize\">\n </div>\n\n <div class=\"bg-white absolute h-[1px] top-0 left-0 right-0\"></div>\n <div class=\"bg-white absolute h-[1px] bottom-0 left-0 right-0\"></div>\n <div class=\"bg-white absolute w-[1px] top-0 left-0 bottom-0\"></div>\n <div class=\"bg-white absolute w-[1px] top-0 bottom-0 right-0\"></div>\n <div class=\"bg-white absolute h-[2px] top-[33%] left-0 right-0\"></div>\n <div class=\"bg-white absolute h-[2px] top-[66%] left-0 right-0\"></div>\n <div class=\"bg-white absolute w-[2px] top-0 bottom-0 left-[33%]\"></div>\n <div class=\"bg-white absolute w-[2px] top-0 bottom-0 left-[66%]\"></div>\n </div>\n </div>\n </div>\n @if (loading()) {\n <libs_ui-components-spinner [size]=\"'medium'\" />\n }\n </div>\n <div class=\"w-[320px] shrink-0 p-[16px] bg-[#ffffff] rounded-br-[8px] z-[2] libs-ui-input-image-editor-container-right\">\n <div class=\"flex items-center justify-between mb-[16px]\">\n <div class=\"libs-ui-font-h4m\">{{ 'i18n_cutting_ratio' | translate }}</div>\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [type]=\"'button-third'\"\n [classIconLeft]=\"'libs-ui-icon-refresh'\"\n [classInclude]=\"'p-[6px] mr-[8px]'\"\n [popover]=\"{config: {content: 'i18n_restore_the_original_state', zIndex: zIndex()}}\"\n [disable]=\"loading() || loadingImage() || !changed()\"\n (outClick)=\"handlerRestoreImage($event)\" />\n </div>\n <div class=\"mb-[16px] libs-ui-image-editor-edit-display-grid-gap-16\">\n @for (item of cropRatioItems(); track item.key) {\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage() || requiredCropFollowRatio()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"item.icon+' !text-[20px]'\"\n [isActive]=\"item.key === cropRatioItemSelected()\"\n (outClick)=\"handlerSelectCropRatioItem($event, item.key)\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ (item.key === 'free' ? 'i18n_custom' : item.key) | translate }}\n </div>\n </div>\n }\n </div>\n <div class=\"mb-[16px] flex\">\n <libs_ui-components-inputs-valid class=\"mr-[16px]\"\n [dataType]=\"'int'\"\n [labelConfig]=\"{labelLeft: 'i18n_width', classInclude: 'mb-[4px]', labelLeftClass: 'libs-ui-font-h7r'}\"\n [fieldNameBind]=\"'width'\"\n [(item)]=\"cropSize\"\n [unitsRight]=\"[{id: 'px', label: 'px'}]\"\n [keySelectedUnitRight]=\"'px'\"\n [valueUpDownNumber]=\"1\"\n [maxValueNumber]=\"originWidth()\"\n [disable]=\"loading() || loadingImage()\"\n (outValueChange)=\"handlerCropWidth()\" />\n <libs_ui-components-inputs-valid [dataType]=\"'int'\"\n [labelConfig]=\"{labelLeft: 'i18n_length', classInclude: 'mb-[4px]', labelLeftClass: 'libs-ui-font-h7r'}\"\n [fieldNameBind]=\"'height'\"\n [(item)]=\"cropSize\"\n [unitsRight]=\"[{id: 'px', label: 'px'}]\"\n [keySelectedUnitRight]=\"'px'\"\n [valueUpDownNumber]=\"1\"\n [maxValueNumber]=\"originHeight()\"\n [disable]=\"loading() || loadingImage()\"\n (outValueChange)=\"handlerCropHeight()\" />\n </div>\n <div class=\"libs-ui-font-h4m mb-[16px]\">{{ 'i18n_rotate_photos' | translate }}</div>\n <div class=\"flex libs-ui-image-editor-edit-display-grid-gap-16\">\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-rotate-image-outline !text-[20px]'\"\n (outClick)=\"handlerRotateImage($event)\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_rotate_ninety_degrees' | translate }}\n </div>\n </div>\n\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-flip-vertical !text-[20px]'\"\n (outClick)=\"handlerFlipImage($event, 'vertical')\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_vertical_flip' | translate }}\n </div>\n </div>\n\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-flip-horizontal !text-[20px]'\"\n (outClick)=\"handlerFlipImage($event, 'horizontal')\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_horizontal_flip' | translate }}\n </div>\n </div>\n\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-scale !text-[20px]'\"\n (outClick)=\"handlerResize($event)\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_resize' | translate }}\n </div>\n </div>\n </div>\n </div>\n </div>\n</libs_ui-components-modal>\n\n<ng-template #svgArrow>\n <svg xmlns=\"http://www.w3.org/2000/svg\"\n width=\"40\"\n height=\"40\"\n viewBox=\"0 0 40 40\"\n fill=\"none\">\n <path d=\"M33.6987 14.6969L18.3309 14.6969C16.516 14.6969 15.0311 16.1818 15.0311 17.9967L15.0311 33.3645\"\n stroke=\"white\"\n stroke-width=\"5.83333\"\n stroke-miterlimit=\"10\"\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n</ng-template>\n\n<ng-template #svgCircle>\n <svg xmlns=\"http://www.w3.org/2000/svg\"\n width=\"15\"\n height=\"15\"\n viewBox=\"0 0 15 15\"\n fill=\"none\">\n <circle cx=\"7.5\"\n cy=\"7.5\"\n r=\"7.5\"\n fill=\"white\" />\n </svg>\n</ng-template>\n", styles: [".libs-ui-image-editor-edit-display-grid-gap-16{display:grid;gap:16px;grid-template-columns:1fr 1fr 1fr 1fr}.libs-ui-image-editor-edit-top-33{top:33.33%}.libs-ui-image-editor-edit-top-66{top:66.66%}.libs-ui-image-editor-edit-left-33{left:33.33%}.libs-ui-image-editor-edit-left-66{left:66.66%}.libs-ui-image-editor-edit-action-item{display:flex;flex-direction:column;align-items:center;width:60px;cursor:pointer}.libs-ui-image-editor-edit-action-item-icon{padding:12px;border-radius:8px;width:max-content;border:1px solid #e6e7ea}.libs-ui-image-editor-edit-action-item-icon:before{font-size:20px}.libs-ui-image-editor-edit-action-item:hover .libs-ui-image-editor-edit-action-item-icon{border:1px solid var(--mo-global-color-primary-border-hover)}.libs-ui-image-editor-edit-action-item:hover .libs-ui-image-editor-edit-action-item-icon:before{color:var(--mo-global-color-primary-text-bright-20)!important}.libs-ui-image-editor-edit-action-item[active=true] .libs-ui-image-editor-edit-action-item-icon{border:1px solid var(--mo-global-color-primary-border-active)!important;background-color:var(--mo-global-color-primary-background-95)!important}.libs-ui-image-editor-edit-action-item[active=true] .libs-ui-image-editor-edit-action-item-icon:before{color:var(--mo-global-color-primary-text-active)!important}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }, { kind: "component", type: LibsUiComponentsPopoverComponent, selector: "libs_ui-components-popover,[LibsUiComponentsPopoverDirective]", inputs: ["debugId", "flagMouse", "type", "mode", "config", "ignoreShowPopover", "elementRefCustom", "initEventInElementRefCustom", "classInclude", "ignoreHiddenPopoverContentWhenMouseLeave", "ignoreStopPropagationEvent", "ignoreCursorPointerModeLikeClick", "isAddContentToParentDocument", "ignoreClickOutside"], outputs: ["outEvent", "outChangStageFlagMouse", "outEventPopoverContent", "outFunctionsControl"] }, { kind: "component", type: LibsUiComponentsModalComponent, selector: "libs_ui-components-modal", inputs: ["show", "mode", "isBackdropTransparent", "isBackgroundTransparentModal", "isSizeBackdropByWidthHeightInput", "hasShadowBoxWhenHiddenBackDropTransparent", "classIncludeModalWrapper", "zIndex", "width", "height", "maxWidth", "maxHeight", "minWidth", "isFullScreen", "disable", "ignoreCommunicateMicroEvent", "headerConfig", "bodyConfig", "footerConfig", "buttonsFooter", "title", "titleUseXssFilter", "titleUseTooltip", "titleUseInnerText"], outputs: ["showChange", "widthChange", "heightChange", "maxWidthChange", "maxHeightChange", "minWidthChange", "disableChange", "buttonsFooterChange", "outScrollContent", "outEvent", "outFunctionControl"] }, { kind: "component", type: LibsUiComponentsInputsValidComponent, selector: "libs_ui-components-inputs-valid", inputs: ["item", "labelConfig", "emitEmptyInDataTypeNumber", "ignoreBlockInputMaxValue", "fieldNameBind", "showCount", "typeComponentSelectItem", "valueComponentSelectItem", "disableComponentSelectItem", "tagInput", "dataType", "typeInput", "modeInput", "resetAutoCompletePassword", "textAreaEnterNotNewLine", "fixedFloat", "acceptNegativeValue", "valueUpDownNumber", "ignoreWidthInput100", "classIncludeInput", "classContainerInput", "readonly", "disable", "noBorder", "backgroundNone", "useColorModeExist", "placeholder", "keepPlaceholderOnly", "classContainerBottomInput", "autoRemoveEmoji", "defaultHeight", "maxHeightTextArea", "minHeightTextArea", "ignoreShowError", "borderError", "iconLeftClass", "popoverContentIconLeft", "iconRightClass", "popoverContentIconRight", "zIndexPopoverContent", "unitsLeft", "configUnitLeft", "keySelectedUnitLeft", "unitsRight", "configUnitRight", "keySelectedUnitRight", "maxValueNumber", "minValueNumber", "ignoreContentLeft", "ignoreContentRight", "isBaselineStyle", "valuePatternShowError", "validPattern", "validRequired", "validMinLength", "validMinValue", "validMaxValue", "validMaxLength", "functionValid", "maxLength", "positionMessageErrorStartInput", "classInclude", "resize", "templateLeftBottomInput", "templateRightBottomInput", "onlyAcceptNegativeValue", "autoAddZeroLessThan10InTypeInt", "maxLengthNumberCount", "classMessageErrorInclude", "ignoreStopPropagationEvent", "ignoreUnitRightClassReadOnly", "paddingRightCustomSpecific", "focusTimeOut", "debounceTimeValidate"], outputs: ["itemChange", "outValueChange", "outSelect", "outIconLeft", "outIconRight", "outClickButtonLabel", "outSwitchEventLabel", "outLabelRightClick", "outEnterInputEvent", "outHeightAreaChange", "outFunctionsControl", "outFocusAndBlur", "outChangeValueByButtonUpDown"] }, { kind: "component", type: LibsUiComponentsButtonsButtonComponent, selector: "libs_ui-components-buttons-button", inputs: ["flagMouse", "type", "buttonCustom", "sizeButton", "label", "disable", "isPending", "imageLeft", "classInclude", "classIconLeft", "classIconRight", "classLabel", "iconOnlyType", "popover", "ignoreStopPropagationEvent", "zIndex", "widthLabelPopover", "styleIconLeft", "styleButton", "ignoreFocusWhenInputTab", "ignoreSetClickWhenShowPopover", "ignorePointerEvent", "isActive", "isHandlerEnterDocumentClickButton"], outputs: ["outClick", "outPopoverEvent", "outFunctionsControl"] }, { kind: "component", type: LibsUiComponentsSpinnerComponent, selector: "libs_ui-components-spinner", inputs: ["type", "size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
768
+ }
769
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LibsUiComponentsImageEditorComponent, decorators: [{
770
+ type: Component,
771
+ args: [{ selector: 'libs_ui-components-image_editor', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [
772
+ NgTemplateOutlet, TranslateModule, LibsUiComponentsPopoverComponent, LibsUiComponentsModalComponent,
773
+ LibsUiComponentsInputsValidComponent, LibsUiComponentsButtonsButtonComponent, LibsUiComponentsSpinnerComponent,
774
+ ], template: "<libs_ui-components-modal [width]=\"'calc(100vw - 400px)'\"\n [height]=\"'calc(100vh - 100px)'\"\n [minWidth]=\"'1200px'\"\n [headerConfig]=\"{ignoreHeaderTheme: true, removeButtonClose: true}\"\n [bodyConfig]=\"{classInclude: '!p-0', scrollOverlayOptions: {scrollX: 'hidden', scrollY: 'hidden'}}\"\n [footerConfig]=\"{hidden: true}\"\n [zIndex]=\"zIndex()\"\n [buttonsFooter]=\"[]\"\n [mode]=\"'center'\">\n <div class=\"libs-ui-modal-header-custom w-full flex items-center justify-between\">\n <div class=\"w-full relative mx-[24px] h-[16px]\">\n <div class=\"flex w-full libs-ui-font-h4s absolute\">\n <libs_ui-components-popover [type]=\"'text'\"\n [config]=\"{zIndex: zIndex()}\">\n {{ 'i18n_edit_file_name' | translate:{file_name: nameFile()} }}\n </libs_ui-components-popover>\n </div>\n </div>\n <div class=\"flex items-center\">\n @if (modeShowButton() === 'save-file') {\n <libs_ui-components-buttons-button [type]=\"'button-third'\"\n [label]=\"'i18n_cancel'\"\n [disable]=\"loading() || loadingImage()\"\n [classInclude]=\"'py-[6px] px-[12px] mr-[16px]'\"\n (outClick)=\"handlerClose($event)\" />\n\n <libs_ui-components-buttons-button class=\"mr-[24px]\"\n [label]=\"'i18n_save'\"\n [disable]=\"loading() || loadingImage()\"\n [classInclude]=\"'py-[6px] px-[12px]'\"\n (outClick)=\"handlerSaveFile($event,'save-file')\" />\n }\n @else {\n <libs_ui-components-buttons-button [label]=\"'i18n_save_to_new_file'\"\n [type]=\"'button-outline'\"\n [classInclude]=\"'mx-[16px] py-[6px] px-[12px]'\"\n [disable]=\"loading() || loadingImage() || !changed()\"\n (outClick)=\"handlerSaveFile($event,'save-api-as-new-file')\" />\n <libs_ui-components-buttons-button [type]=\"'button-secondary'\"\n [label]=\"'i18n_save'\"\n [classInclude]=\"'py-[6px] px-[12px]'\"\n [disable]=\"loading() || loadingImage() || !changed()\"\n (outClick)=\"handlerSaveFile($event,'save-api')\" />\n <div class=\"h-[12px] libs-ui-border-left-general mx-[16px]\"></div>\n <libs_ui-components-buttons-button class=\"mr-[24px]\"\n [classInclude]=\"'p-[6px]'\"\n [type]=\"'button-third-hover-danger'\"\n [iconOnlyType]=\"true\"\n [disable]=\"loading()\"\n [popover]=\"{config: {content: 'i18n_exit_edit_mode', zIndex: zIndex()}}\"\n [classIconLeft]=\"'libs-ui-icon-close'\"\n (outClick)=\"handlerClose($event)\" />\n }\n </div>\n </div>\n <div class=\"libs-ui-modal-body-custom h-full w-full flex\">\n <div #imageEditorContainer\n class=\"relative w-full h-full bg-[#e6e7ea] border-radius-bottom-left-8px p-[32px] libs-ui-input-image-editor-container-left\">\n @if (loadingImage()) {\n <div class=\"absolute h-full w-full bg-[#e6e7ea] top-0 left-0 z-[5]\">\n <libs_ui-components-spinner [size]=\"'medium'\" />\n </div>\n }\n <div class=\"absolute h-[32px] w-[calc(100%-64px)] bg-[#e6e7eab8] top-0 left-[32px] z-[1]\">\n </div>\n <div class=\"absolute w-[32px] h-full bg-[#e6e7eab8] left-0 top-0 z-[1] rounded-bl-[8px]\">\n </div>\n <div class=\"absolute w-[32px] h-full bg-[#e6e7eab8] right-0 top-0 z-[1]\">\n </div>\n <div class=\"absolute h-[32px] w-[calc(100%-64px)] bg-[#e6e7eab8] bottom-0 left-[32px] z-[1]\">\n </div>\n <div #imageContainer\n class=\"relative w-full h-full rounded-bl-[8px] min-h-0\">\n <div #circleTL\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <div #circleTR\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <div #circleBL\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <div #circleBR\n class=\"absolute z-[1]\"\n [class.hidden]=\"!dragGridCrop()\"\n [class.flex]=\"dragGridCrop()\">\n <ng-template *ngTemplateOutlet=\"svgCircle\" />\n </div>\n <img #imageOrigin\n class=\"absolute border-[4px] border-[#8e61ee] rounded-[4px] top-[-1000px] left-[-1000px]\"\n loading=\"lazy\"\n [src]=\"imgSrc()\"\n (load)=\"handlerImageLoaded()\"\n (error)=\"handlerImageError($event)\" />\n <img #imageClip\n class=\"absolute z-[1] top-[-1000px] left-[-1000px]\"\n loading=\"lazy\"\n [src]=\"imgSrc()\"\n (error)=\"handlerImageError($event)\" />\n <div class=\"absolute top-0 left-0 right-0 bottom-0 bg-[#e6e7eab8]\">\n </div>\n <div #cropArea\n class=\"absolute z-[2] border-[4px] border-[#8e61ee] rounded-[4px]\">\n <div class=\"relative w-full h-full\">\n <div #cropTL\n class=\"absolute z-[2] left-[-18px] top-[-17px] cursor-nwse-resize\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropBL\n class=\"absolute z-[2] left-[-17px] bottom-[-18px] cursor-nesw-resize rotate-[270deg]\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropTR\n class=\"absolute z-[2] right-[-17px] top-[-17px] cursor-nesw-resize rotate-[90deg]\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropBR\n class=\"absolute z-[2] right-[-18px] bottom-[-17px] cursor-nwse-resize rotate-[180deg]\">\n <ng-template *ngTemplateOutlet=\"svgArrow\" />\n </div>\n <div #cropLineVL\n class=\"absolute z-[1] w-[12px] left-[-6px] top-0 bottom-0 cursor-ew-resize\">\n </div>\n <div #cropLineVR\n class=\"absolute z-[1] w-[12px] right-[-6px] top-0 bottom-0 cursor-ew-resize\">\n </div>\n <div #cropLineHT\n class=\"absolute z-[1] h-[12px] left-0 top-[-6px] right-0 cursor-ns-resize\">\n </div>\n <div #cropLineHB\n class=\"absolute z-[1] h-[12px] left-0 right-0 bottom-[-6px] cursor-ns-resize\">\n </div>\n\n <div class=\"bg-white absolute h-[1px] top-0 left-0 right-0\"></div>\n <div class=\"bg-white absolute h-[1px] bottom-0 left-0 right-0\"></div>\n <div class=\"bg-white absolute w-[1px] top-0 left-0 bottom-0\"></div>\n <div class=\"bg-white absolute w-[1px] top-0 bottom-0 right-0\"></div>\n <div class=\"bg-white absolute h-[2px] top-[33%] left-0 right-0\"></div>\n <div class=\"bg-white absolute h-[2px] top-[66%] left-0 right-0\"></div>\n <div class=\"bg-white absolute w-[2px] top-0 bottom-0 left-[33%]\"></div>\n <div class=\"bg-white absolute w-[2px] top-0 bottom-0 left-[66%]\"></div>\n </div>\n </div>\n </div>\n @if (loading()) {\n <libs_ui-components-spinner [size]=\"'medium'\" />\n }\n </div>\n <div class=\"w-[320px] shrink-0 p-[16px] bg-[#ffffff] rounded-br-[8px] z-[2] libs-ui-input-image-editor-container-right\">\n <div class=\"flex items-center justify-between mb-[16px]\">\n <div class=\"libs-ui-font-h4m\">{{ 'i18n_cutting_ratio' | translate }}</div>\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [type]=\"'button-third'\"\n [classIconLeft]=\"'libs-ui-icon-refresh'\"\n [classInclude]=\"'p-[6px] mr-[8px]'\"\n [popover]=\"{config: {content: 'i18n_restore_the_original_state', zIndex: zIndex()}}\"\n [disable]=\"loading() || loadingImage() || !changed()\"\n (outClick)=\"handlerRestoreImage($event)\" />\n </div>\n <div class=\"mb-[16px] libs-ui-image-editor-edit-display-grid-gap-16\">\n @for (item of cropRatioItems(); track item.key) {\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage() || requiredCropFollowRatio()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"item.icon+' !text-[20px]'\"\n [isActive]=\"item.key === cropRatioItemSelected()\"\n (outClick)=\"handlerSelectCropRatioItem($event, item.key)\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ (item.key === 'free' ? 'i18n_custom' : item.key) | translate }}\n </div>\n </div>\n }\n </div>\n <div class=\"mb-[16px] flex\">\n <libs_ui-components-inputs-valid class=\"mr-[16px]\"\n [dataType]=\"'int'\"\n [labelConfig]=\"{labelLeft: 'i18n_width', classInclude: 'mb-[4px]', labelLeftClass: 'libs-ui-font-h7r'}\"\n [fieldNameBind]=\"'width'\"\n [(item)]=\"cropSize\"\n [unitsRight]=\"[{id: 'px', label: 'px'}]\"\n [keySelectedUnitRight]=\"'px'\"\n [valueUpDownNumber]=\"1\"\n [maxValueNumber]=\"originWidth()\"\n [disable]=\"loading() || loadingImage()\"\n (outValueChange)=\"handlerCropWidth()\" />\n <libs_ui-components-inputs-valid [dataType]=\"'int'\"\n [labelConfig]=\"{labelLeft: 'i18n_length', classInclude: 'mb-[4px]', labelLeftClass: 'libs-ui-font-h7r'}\"\n [fieldNameBind]=\"'height'\"\n [(item)]=\"cropSize\"\n [unitsRight]=\"[{id: 'px', label: 'px'}]\"\n [keySelectedUnitRight]=\"'px'\"\n [valueUpDownNumber]=\"1\"\n [maxValueNumber]=\"originHeight()\"\n [disable]=\"loading() || loadingImage()\"\n (outValueChange)=\"handlerCropHeight()\" />\n </div>\n <div class=\"libs-ui-font-h4m mb-[16px]\">{{ 'i18n_rotate_photos' | translate }}</div>\n <div class=\"flex libs-ui-image-editor-edit-display-grid-gap-16\">\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-rotate-image-outline !text-[20px]'\"\n (outClick)=\"handlerRotateImage($event)\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_rotate_ninety_degrees' | translate }}\n </div>\n </div>\n\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-flip-vertical !text-[20px]'\"\n (outClick)=\"handlerFlipImage($event, 'vertical')\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_vertical_flip' | translate }}\n </div>\n </div>\n\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-flip-horizontal !text-[20px]'\"\n (outClick)=\"handlerFlipImage($event, 'horizontal')\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_horizontal_flip' | translate }}\n </div>\n </div>\n\n <div class=\"flex flex-col items-center\">\n <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n [sizeButton]=\"'large'\"\n [disable]=\"loading() || loadingImage()\"\n [type]=\"'button-outline'\"\n [classIconLeft]=\"'libs-ui-icon-scale !text-[20px]'\"\n (outClick)=\"handlerResize($event)\" />\n <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n {{ 'i18n_resize' | translate }}\n </div>\n </div>\n </div>\n </div>\n </div>\n</libs_ui-components-modal>\n\n<ng-template #svgArrow>\n <svg xmlns=\"http://www.w3.org/2000/svg\"\n width=\"40\"\n height=\"40\"\n viewBox=\"0 0 40 40\"\n fill=\"none\">\n <path d=\"M33.6987 14.6969L18.3309 14.6969C16.516 14.6969 15.0311 16.1818 15.0311 17.9967L15.0311 33.3645\"\n stroke=\"white\"\n stroke-width=\"5.83333\"\n stroke-miterlimit=\"10\"\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n</ng-template>\n\n<ng-template #svgCircle>\n <svg xmlns=\"http://www.w3.org/2000/svg\"\n width=\"15\"\n height=\"15\"\n viewBox=\"0 0 15 15\"\n fill=\"none\">\n <circle cx=\"7.5\"\n cy=\"7.5\"\n r=\"7.5\"\n fill=\"white\" />\n </svg>\n</ng-template>\n", styles: [".libs-ui-image-editor-edit-display-grid-gap-16{display:grid;gap:16px;grid-template-columns:1fr 1fr 1fr 1fr}.libs-ui-image-editor-edit-top-33{top:33.33%}.libs-ui-image-editor-edit-top-66{top:66.66%}.libs-ui-image-editor-edit-left-33{left:33.33%}.libs-ui-image-editor-edit-left-66{left:66.66%}.libs-ui-image-editor-edit-action-item{display:flex;flex-direction:column;align-items:center;width:60px;cursor:pointer}.libs-ui-image-editor-edit-action-item-icon{padding:12px;border-radius:8px;width:max-content;border:1px solid #e6e7ea}.libs-ui-image-editor-edit-action-item-icon:before{font-size:20px}.libs-ui-image-editor-edit-action-item:hover .libs-ui-image-editor-edit-action-item-icon{border:1px solid var(--mo-global-color-primary-border-hover)}.libs-ui-image-editor-edit-action-item:hover .libs-ui-image-editor-edit-action-item-icon:before{color:var(--mo-global-color-primary-text-bright-20)!important}.libs-ui-image-editor-edit-action-item[active=true] .libs-ui-image-editor-edit-action-item-icon{border:1px solid var(--mo-global-color-primary-border-active)!important;background-color:var(--mo-global-color-primary-background-95)!important}.libs-ui-image-editor-edit-action-item[active=true] .libs-ui-image-editor-edit-action-item-icon:before{color:var(--mo-global-color-primary-text-active)!important}\n"] }]
775
+ }], ctorParameters: () => [{ type: undefined, decorators: [{
776
+ type: Optional
777
+ }, {
778
+ type: Inject,
779
+ args: [LINK_IMAGE_ERROR_TOKEN_INJECT]
780
+ }] }] });
781
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-editor.component.js","sourceRoot":"","sources":["../../../../../libs-ui/components/image-editor/src/image-editor.component.ts","../../../../../libs-ui/components/image-editor/src/image-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { NgTemplateOutlet } from '@angular/common';\nimport { AfterViewInit, ChangeDetectionStrategy, Component, ComponentRef, DestroyRef, ElementRef, Inject, inject, input, model, OnDestroy, Optional, output, signal, viewChild } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { LibsUiComponentsButtonsButtonComponent } from '@libs-ui/components-buttons-button';\nimport { LibsUiComponentsInputsValidComponent } from '@libs-ui/components-inputs-valid';\nimport { LibsUiComponentsModalComponent } from '@libs-ui/components-modal';\nimport { LibsUiComponentsPopoverComponent } from '@libs-ui/components-popover';\nimport { LibsUiComponentsSpinnerComponent } from '@libs-ui/components-spinner';\nimport { IAspectRatio, IBoundingClientRect, IEvent } from '@libs-ui/interfaces-types';\nimport { LibsUiDynamicComponentService } from '@libs-ui/services-dynamic-component';\nimport { convertBase64ToBlob, get, isNil, LINK_IMAGE_ERROR_TOKEN_INJECT, set } from '@libs-ui/utils';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { fromEvent, Observable } from 'rxjs';\nimport { mergeMap, takeUntil, tap } from 'rxjs/operators';\nimport { cropRationItems, getCropRectImage, getDataUrl, getStylesOfElement, getWidthHeightResizeCropFollow } from './defines/image-editor.define';\nimport { IImageEditorFunctionControlEvent } from './interfaces/function-control-event.interface';\nimport { ICropRatioItem, ICropSize, IRectClip, IResizeData, ISaveFile, IStartMouseDim, IStyles, MODE_EDIT, TYPE_MODE_SAVE, TYPE_RESIZE_STATE } from './interfaces/image-editor.interface';\nimport { LibsUiComponentsImageEditorResizeComponent } from './resize/resize.component';\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'libs_ui-components-image_editor',\n  templateUrl: './image-editor.component.html',\n  styleUrl: './image-editor.component.scss',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    NgTemplateOutlet, TranslateModule, LibsUiComponentsPopoverComponent, LibsUiComponentsModalComponent,\n    LibsUiComponentsInputsValidComponent, LibsUiComponentsButtonsButtonComponent, LibsUiComponentsSpinnerComponent,\n  ]\n})\nexport class LibsUiComponentsImageEditorComponent implements AfterViewInit, OnDestroy {\n  protected loading = signal(false);\n  protected loadingImage = signal(true);\n  protected changed = signal(false);\n  protected originWidth = signal<number>(0);\n  protected originHeight = signal<number>(0);\n  protected cropRatioItems = signal<Array<ICropRatioItem>>(cropRationItems());\n  protected cropRatioItemSelected = signal<string>('');\n  protected cropSize = signal<ICropSize>({ width: 20, height: 20 });\n  protected resizeData = signal<IResizeData>({ ratio: 100, width: 600, height: 800 });\n  protected dragGridCrop = signal(false);\n\n  private image = new Image();\n  private canvas = document.createElement('canvas');\n  private containerHeight = signal(800);\n  private containerWidth = signal(1200);\n  private imgContainerRect = signal<IBoundingClientRect>({ top: 0, left: 0, width: 0, height: 0 });\n  private rectClip = signal<IRectClip>({ top: 0, left: 0, right: 0, bottom: 0, width: 0, height: 0 });\n  private resizeState = signal<TYPE_RESIZE_STATE>('none');\n  private moveState = signal<'clip' | 'image' | 'none'>('none');\n  private startMouseDim = signal<IStartMouseDim>({ clientX: 0, clientY: 0, width: 0, height: 0, top: 0, left: 0, imageTop: 0, imageLeft: 0, imageWidth: 0, imageHeight: 0 });\n  private dataUrlOrigin = signal<string>('');\n  private ratioValue = signal<number>(0);\n  private readonly minHeight = 20;\n  private readonly minWidth = 20;\n  private resizeComponentRef?: ComponentRef<LibsUiComponentsImageEditorResizeComponent>;\n\n\n  readonly modeShowButton = input<MODE_EDIT>('save-file');\n  readonly mimetype = input<string>();\n  readonly zIndex = input<number>(1200);\n  readonly imgSrc = model.required<string>();\n  readonly originUrl = input<string>();\n  readonly nameFile = input<string>();\n  readonly hasZoom = input<boolean>(false);\n  readonly aspectRatio = input<IAspectRatio>();\n  readonly requiredCropFollowRatio = input<boolean>(false);\n\n  readonly imageOrigin = viewChild.required<ElementRef>('imageOrigin');\n  readonly imageClip = viewChild.required<ElementRef>('imageClip');\n  readonly imageEditorContainer = viewChild.required<ElementRef>('imageEditorContainer');\n  readonly imageContainer = viewChild.required<ElementRef>('imageContainer');\n  readonly cropArea = viewChild.required<ElementRef>('cropArea');\n  readonly cropTL = viewChild.required<ElementRef>('cropTL');\n  readonly cropBL = viewChild.required<ElementRef>('cropBL');\n  readonly cropBR = viewChild.required<ElementRef>('cropBR');\n  readonly cropTR = viewChild.required<ElementRef>('cropTR');\n  readonly cropLineVL = viewChild.required<ElementRef>('cropLineVL');\n  readonly cropLineVR = viewChild.required<ElementRef>('cropLineVR');\n  readonly cropLineHT = viewChild.required<ElementRef>('cropLineHT');\n  readonly cropLineHB = viewChild.required<ElementRef>('cropLineHB');\n  readonly circleTL = viewChild.required<ElementRef>('circleTL');\n  readonly circleTR = viewChild.required<ElementRef>('circleTR');\n  readonly circleBL = viewChild.required<ElementRef>('circleBL');\n  readonly circleBR = viewChild.required<ElementRef>('circleBR');\n\n  readonly outClose = output<{ isClickButtonClose: boolean }>();\n  readonly outSaveFile = output<ISaveFile>();\n  readonly outFunctionsControl = output<IImageEditorFunctionControlEvent>();\n\n  private destroyRef = inject(DestroyRef)\n  private dynamicComponentService = inject(LibsUiDynamicComponentService);\n\n  constructor(@Optional() @Inject(LINK_IMAGE_ERROR_TOKEN_INJECT) private linkImageError: string) {\n  }\n\n  ngAfterViewInit() {\n    const aspectRatioFound = this.aspectRatio()?.key ? this.cropRatioItems().find(item => item.key === this.aspectRatio()?.key) : this.cropRatioItems()[0];\n\n    if (aspectRatioFound) {\n      this.dragGridCrop.set(true);\n\n      this.cropRatioItemSelected.set(aspectRatioFound?.key);\n      this.updateOriginImageSize();\n      this.changedImage(true);\n    }\n\n    this.updateModalSize();\n    this.initData();\n    this.initMouseEvent();\n    this.outFunctionsControl.emit({\n      cropImage: this.cropImage.bind(this),\n      setLoadingState: (loading: boolean) => this.loading.set(loading)\n    });\n  }\n\n  protected handlerImageLoaded() {\n    setTimeout(() => {\n      this.updateModalSize();\n      this.loadingImage.set(false);\n    }, 500);\n  }\n\n  protected handlerImageError(e: ErrorEvent) {\n    console.log('e', e);\n    (e as IEvent).target.src = this.linkImageError;\n  }\n\n  private updateModalSize() {\n    const maxHeight = window.innerHeight - 100;\n    const maxWidth = window.innerWidth - 400;\n\n    this.containerHeight.set(maxHeight - 124);\n    this.containerWidth.set(maxWidth - 384);\n    this.updateOriginImageSize();\n  }\n\n  private initData() {\n    this.imgContainerRect.set(this.imageContainer().nativeElement.getBoundingClientRect());\n    this.image.setAttribute('crossorigin', 'anonymous');\n    this.image.onload = () => {\n      this.imgContainerRect.set(this.imageContainer().nativeElement.getBoundingClientRect());\n      this.canvas.height = this.image.height;\n      this.canvas.width = this.image.width;\n      this.originHeight.set(this.image.height);\n      this.originWidth.set(this.image.width);\n\n      this.updateOriginImageSize();\n      this.canvas.getContext('2d')?.drawImage(this.image, 0, 0, this.canvas.width, this.canvas.height);\n      this.dataUrlOrigin.set(getDataUrl(this.canvas, this.mimetype(), this.image.src));\n      this.imgSrc.set(this.dataUrlOrigin());\n    };\n    this.image.src = this.imgSrc();\n  }\n\n  private initMouseEvent() {\n    const mouseUp = this.initEvent({ nativeElement: document }, 'mouseup', { callStopPropagation: true, callPreventDefault: true }, undefined, () => this.handlerMouseup());\n    const mouseMove = this.initEvent({ nativeElement: document }, 'mousemove', { callStopPropagation: true, callPreventDefault: true }).pipe(takeUntil(mouseUp));\n\n    this.initEvent(this.imageEditorContainer(), 'wheel', { callStopPropagation: true }, undefined, (e: Event) => this.handlerMousewheel(e)).subscribe();\n    this.initEvent({ nativeElement: window }, 'resize', {}, undefined, () => this.updateModalSize()).subscribe();\n    this.initEvent(this.imageContainer(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => this.handlerImageContainerMousedown(e)).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropTL(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'TL') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropBL(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'BL') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropBR(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'BR') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropTR(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'TR') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropLineVL(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'VL') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropLineVR(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'VR') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropLineHT(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'HT') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropLineHB(), 'mousedown', { callStopPropagation: true, callPreventDefault: true }, mouseMove, (e: Event) => { this.handlerCropResize(e, 'HB') }).subscribe((e: Event) => this.handlerMousemove(e));\n    this.initEvent(this.cropArea(), 'mousedown', {}, undefined, (e: Event) => this.handlerMousedownCropArea(e)).subscribe((e: Event) => this.handlerMousemove(e));\n  }\n\n  private initEvent(element: ElementRef, eventName: string, flag: { callStopPropagation?: boolean, callPreventDefault?: boolean }, obsMerge?: Observable<Event>, callback?: (e: Event) => void): Observable<Event> {\n    const obs = fromEvent<Event>(element.nativeElement, eventName).pipe(\n      tap((e: Event) => {\n        if (flag.callStopPropagation && e?.stopPropagation) {\n          e.stopPropagation();\n        }\n        if (flag.callPreventDefault && e?.preventDefault) {\n          e.preventDefault();\n        }\n        if (callback) {\n          callback(e);\n        }\n      }),\n      takeUntilDestroyed(this.destroyRef));\n\n    if (obsMerge) {\n      return obs.pipe(mergeMap(() => obsMerge), (takeUntilDestroyed(this.destroyRef)));\n    }\n\n    return obs;\n  };\n\n  private updateOriginImageSize() {\n    let width = this.originWidth();\n    let height = this.originHeight();\n    const ratio = width / height;\n\n    if (width >= height) {\n      width = Math.min(this.containerWidth(), width);\n      height = Math.min(width / ratio, this.containerHeight());\n      width = height * ratio;\n    } else {\n      height = Math.min(this.containerHeight(), height);\n      width = height * ratio;\n    }\n    this.setImagePosition(width, height);\n    this.updateCropAreaPos();\n    this.updateRectClipPos();\n  }\n\n  private setImagePosition(width: number, height: number) {\n    const left = this.containerWidth() / 2 - width / 2;\n    const top = this.containerHeight() / 2 - height / 2;\n    const styles = { width, height, top, left, display: 'block' };\n\n    this.setStylesElements(this.imageOrigin(), styles);\n    this.setStylesElements(this.imageClip(), styles);\n    this.setCirclePosition(styles);\n  }\n\n  private setCirclePosition(styles: IStyles) {\n    this.setStylesElements(this.circleTL(), { top: (styles.top || 0) - 5, left: (styles.left || 0) - 5 });\n    this.setStylesElements(this.circleTR(), { top: (styles.top || 0) - 5, left: (styles.left || 0) + (styles.width || 0) - 10 });\n    this.setStylesElements(this.circleBL(), { top: (styles.top || 0) + (styles.height || 0) - 10, left: (styles.left || 0) - 5 });\n    this.setStylesElements(this.circleBR(), { top: (styles.top || 0) + (styles.height || 0) - 10, left: (styles.left || 0) + (styles.width || 0) - 10 });\n  }\n\n  private updateCropAreaPos() {\n    const [imgTop, imgLeft, imgWidth, imgHeight] = getStylesOfElement<number>(this.imageClip().nativeElement, ['style.top', 'style.left', 'offsetWidth', 'offsetHeight']);\n\n    const rectImg: IBoundingClientRect = {\n      top: Math.max(0, imgTop),\n      left: Math.max(0, imgLeft),\n      width: imgWidth,\n      height: imgHeight\n    };\n\n    rectImg.width = Math.min(imgWidth, this.containerWidth() - imgLeft);\n    if (imgLeft < 0) {\n      rectImg.width = Math.min(imgWidth + imgLeft, this.containerWidth());\n    }\n    rectImg.height = Math.min(imgHeight, this.containerHeight() - imgTop);\n    if (imgTop < 0) {\n      rectImg.height = Math.min(imgTop + imgHeight, this.containerHeight());\n    }\n    this.ratioValue.set(this.getCropRatioValue());\n    if (!this.ratioValue()) {\n      this.setStylesElements(this.cropArea(), rectImg);\n\n      return;\n    }\n    let width = rectImg.width;\n    let height = width / this.ratioValue();\n\n    width = Math.min(rectImg.width, width);\n    height = width / this.ratioValue();\n    height = Math.min(rectImg.height, height);\n    width = height * this.ratioValue();\n    const top = rectImg.top + (rectImg.height - height) / 2;\n    const left = rectImg.left + (rectImg.width - width) / 2;\n\n    this.setStylesElements(this.cropArea(), { width, height, top, left });\n  }\n\n  private getCropRatioValue() {\n    const cropRatio = this.cropRatioItems().find(item => item.key === this.cropRatioItemSelected());\n\n    if (!cropRatio?.value) {\n      return this.aspectRatio()?.value || (this.originWidth() / this.originHeight());\n    }\n\n    return cropRatio.value;\n  }\n\n  private updateRectClipPos() {\n    const reactClip = this.getRectClipImage();\n\n    this.imageClip().nativeElement.style.clip = `rect(${reactClip.top}px, ${reactClip.right}px, ${reactClip.bottom}px, ${reactClip.left}px)`;\n  }\n\n  private getRectClipImage() {\n\n    const imageRect = this.imageClip().nativeElement.getBoundingClientRect();\n    if (!imageRect.width) {\n\n      return this.rectClip();\n    }\n    const cropRect = this.cropArea().nativeElement.getBoundingClientRect();\n    const width = Math.round(cropRect.width);\n    const height = Math.round(cropRect.height);\n    const left = cropRect.left - imageRect.left;\n    const top = cropRect.top - imageRect.top;\n    const right = Math.round(left + width);\n    const bottom = Math.round(top + height);\n    const ratio = this.originWidth() / imageRect.width;\n\n\n    this.rectClip.set({ left, top, right, bottom, width, height });\n    this.cropSize.set({\n      height: top >= 0 ? (Math.round((bottom - top) * ratio)) : (Math.floor((bottom - top) * ratio)),\n      width: top >= 0 ? (Math.round(((right - left) * (this.ratioValue() ?? 1)) * ratio)) : (Math.floor(((right - left) * (this.ratioValue() ?? 1)) * ratio))\n    });\n\n    return this.rectClip();\n  }\n\n  private handlerMousemove(e: any) {\n    const typeState = ['BL', 'TL', 'TR', 'BR', 'VL', 'VR', 'HT', 'HB'];\n    const methodName = `resizeCropFollow${this.resizeState()}Dir`; // resizeCropFollowBLDir, resizeCropFollowTLDir, resizeCropFollowTRDir, resizeCropFollowBRDir, resizeCropFollowVLDir, resizeCropFollowVRDir, resizeCropFollowHTDir, resizeCropFollowHBDir\n\n    if (typeState.includes(this.resizeState())) {\n      (this as any)[methodName](e);\n    }\n    switch (this.moveState()) {\n      case 'clip':\n        this.moveClipArea(e);\n        break;\n\n      case 'image':\n        this.moveImage(e);\n        break;\n    }\n    this.changedImage(true);\n  }\n\n  protected resizeCropFollowBLDir = (e: IEvent) => {\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = this.ratioValue() ? -momentY * this.ratioValue() : e.clientX - this.startMouseDim().clientX;\n    const maxHeight = this.startMouseDim().imageTop + this.startMouseDim().imageHeight - this.startMouseDim().top;\n    const maxWidth = this.startMouseDim().left + this.startMouseDim().width - this.startMouseDim().imageLeft;\n    let { width, height, left } = this.getRectResizeCropFollow({ width: -1, height: 1, left: 1, top: 0 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    left = this.startMouseDim().left + this.startMouseDim().width - width;\n    this.setStylesElements(this.cropArea(), { left, height, width }, true);\n  };\n\n  protected resizeCropFollowTLDir = (e: IEvent) => {\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = this.ratioValue() ? this.ratioValue() * momentY : e.clientX - this.startMouseDim().clientX;\n    const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;\n    const maxWidth = this.startMouseDim().left + this.startMouseDim().width - this.startMouseDim().imageLeft;\n    let { width, height, top, left } = this.getRectResizeCropFollow({ width: -1, height: -1, left: 1, top: 1 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    left = this.startMouseDim().left + this.startMouseDim().width - width;\n    top = this.startMouseDim().top + this.startMouseDim().height - height;\n    this.setStylesElements(this.cropArea(), { top, left, height, width }, true);\n  };\n\n  protected resizeCropFollowTRDir = (e: IEvent) => {\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = this.ratioValue() ? -momentY * this.ratioValue() : e.clientX - this.startMouseDim().clientX;\n    const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;\n    const maxWidth = this.startMouseDim().imageLeft + this.startMouseDim().imageWidth - this.startMouseDim().left;\n    let { width, height, top } = this.getRectResizeCropFollow({ width: 1, height: -1, left: 0, top: 1 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    top = this.startMouseDim().top + this.startMouseDim().height - height;\n    this.setStylesElements(this.cropArea(), { top, height, width }, true);\n  };\n\n  protected resizeCropFollowBRDir(e: IEvent) {\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = this.ratioValue() ? momentY * this.ratioValue() : e.clientX - this.startMouseDim().clientX;\n    const maxHeight = this.startMouseDim().imageTop + this.startMouseDim().imageHeight - this.startMouseDim().top;\n    const maxWidth = this.startMouseDim().imageLeft + this.startMouseDim().imageWidth - this.startMouseDim().left;\n    let { width, height } = this.getRectResizeCropFollow({ width: 1, height: 1, left: 0, top: 0 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    this.setStylesElements(this.cropArea(), { height, width }, true);\n  }\n\n  protected resizeCropFollowVLDir = (e: IEvent) => {\n    const momentX = e.clientX - this.startMouseDim().clientX;\n    const momentY = this.ratioValue() ? momentX / this.ratioValue() : 0;\n    const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;\n    const maxWidth = this.startMouseDim().left + this.startMouseDim().width - this.startMouseDim().imageLeft;\n    let { width, height, top, left } = this.getRectResizeCropFollow({ width: -1, height: -1, left: 1, top: 1 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    left = this.startMouseDim().left + this.startMouseDim().width - width;\n    top = this.startMouseDim().top + this.startMouseDim().height - height;\n    this.setStylesElements(this.cropArea(), { top, left, height, width }, true);\n  };\n\n  protected resizeCropFollowVRDir = (e: IEvent) => {\n    const momentX = e.clientX - this.startMouseDim().clientX;\n    const momentY = this.ratioValue() ? -momentX / this.ratioValue() : 0;\n    const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;\n    const maxWidth = this.startMouseDim().imageLeft + this.startMouseDim().imageWidth - this.startMouseDim().left;\n    let { width, height, top } = this.getRectResizeCropFollow({ width: 1, height: -1, left: 0, top: 1 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    top = this.startMouseDim().top + this.startMouseDim().height - height;\n    this.setStylesElements(this.cropArea(), { top, height, width }, true);\n  };\n\n  protected resizeCropFollowHTDir = (e: IEvent) => {\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = this.ratioValue() ? momentY * this.ratioValue() : 0;\n    const maxWidth = this.startMouseDim().width + this.startMouseDim().left - this.startMouseDim().imageLeft;\n    const maxHeight = this.startMouseDim().top + this.startMouseDim().height - this.startMouseDim().imageTop;\n    let { width, height, top, left } = this.getRectResizeCropFollow({ width: -1, height: -1, left: 1, top: 1 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    top = this.startMouseDim().top + this.startMouseDim().height - height;\n    left = this.startMouseDim().left + this.startMouseDim().width - width;\n    this.setStylesElements(this.cropArea(), { top, left, height, width }, true);\n  };\n\n  protected resizeCropFollowHBDir = (e: IEvent) => {\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = this.ratioValue() ? momentY * this.ratioValue() : 0;\n    const maxHeight = this.startMouseDim().imageTop + this.startMouseDim().imageHeight - this.startMouseDim().top;\n    const maxWidth = this.startMouseDim().width + this.startMouseDim().left - this.startMouseDim().imageLeft;\n    let { width, height, left } = this.getRectResizeCropFollow({ width: 1, height: 1, left: -1, top: 0 }, momentX, momentY);\n\n    [width, height] = getWidthHeightResizeCropFollow(this.ratioValue(), width, height, maxWidth, maxHeight, this.minWidth, this.minHeight);\n    left = this.startMouseDim().left + this.startMouseDim().width - width;\n    this.setStylesElements(this.cropArea(), { height, left, width }, true);\n  };\n\n  private getRectResizeCropFollow(ratio: { width: number; height: number; top: number; left: number }, momentX: number, momentY: number) {\n    const width = this.startMouseDim().width + momentX * ratio.width;\n    const height = this.startMouseDim().height + momentY * ratio.height;\n    const top = this.startMouseDim().top + momentY * ratio.top;\n    const left = this.startMouseDim().left + momentX * ratio.left;\n\n    return { width, height, left, top };\n  }\n\n  private handlerMouseup() {\n    this.imageContainer().nativeElement.classList.remove('cursor-grabbing');\n    this.cropArea().nativeElement.classList.remove('cursor-grabbing');\n    this.resizeState.set('none');\n    this.moveState.set('none');\n  }\n\n  private handlerMousewheel(event: any) {\n    if (!this.hasZoom()) {\n      return;\n    }\n    if (event.deltaY < 0) {\n      this.zoomImage(1 / 1.1);\n\n      return;\n    }\n    this.zoomImage(1.1);\n  }\n\n  private handlerImageContainerMousedown(e: any) {\n    if (this.resizeState() === 'none' && this.moveState() === 'none') {\n      this.moveState.set('image');\n      this.handlerMousedown(e);\n    }\n\n    return false;\n  }\n\n  private handlerCropResize(e: any, resizeType: TYPE_RESIZE_STATE) {\n    this.resizeState.set(resizeType);\n    this.dragGridCrop.set(true);\n    this.handlerMousedown(e);\n  }\n\n  private handlerMousedownCropArea(e: any) {\n    this.moveState.set('clip');\n    this.handlerMousedown(e);\n  }\n\n  private handlerMousedown(e: IEvent) {\n    const [width, height, top, left] = getStylesOfElement<number>(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);\n    const [imageWidth, imageHeight, imageTop, imageLeft] = getStylesOfElement<number>(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);\n    switch (this.moveState()) {\n      case 'image':\n        this.imageContainer().nativeElement.classList.add('cursor-grabbing');\n        break;\n      case 'clip':\n        if (width === imageWidth && height === imageHeight) {\n          return;\n        }\n        this.cropArea().nativeElement.classList.add('cursor-grabbing');\n        break;\n\n      default:\n        break;\n    }\n\n    this.startMouseDim.set({\n      clientX: e.clientX,\n      clientY: e.clientY,\n      width: width,\n      height: height,\n      top: top,\n      left: left,\n      imageTop: imageTop,\n      imageLeft: imageLeft,\n      imageWidth: imageWidth,\n      imageHeight: imageHeight\n    });\n    this.ratioValue.set(this.getCropRatioValue());\n  }\n\n  private moveImage(e: IEvent) {\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = e.clientX - this.startMouseDim().clientX;\n    const imgTop = this.startMouseDim().imageTop + momentY;\n    const imgLeft = this.startMouseDim().imageLeft + momentX;\n    const top = this.startMouseDim().top + momentY;\n    const left = this.startMouseDim().left + momentX;\n\n\n    this.setStylesElements(this.imageOrigin(), { top: imgTop, left: imgLeft });\n    this.setStylesElements(this.imageClip(), { top: imgTop, left: imgLeft });\n    this.setStylesElements(this.cropArea(), { top, left });\n    this.setCirclePosition({ top: imgTop, left: imgLeft, width: this.startMouseDim().imageWidth, height: this.startMouseDim().imageHeight });\n  }\n\n  private moveClipArea(e: IEvent) {\n    const [imgWidth, imgHeight, imgTop, imgLeft] = getStylesOfElement<number>(this.imageClip().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);\n    const [cropWidth, cropHeight] = getStylesOfElement<number>(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight']);\n    const momentY = e.clientY - this.startMouseDim().clientY;\n    const momentX = e.clientX - this.startMouseDim().clientX;\n    let top = Math.max(imgTop, this.startMouseDim().top + momentY);\n    let left = Math.max(imgLeft, this.startMouseDim().left + momentX);\n\n    top = Math.min(top, imgTop + imgHeight - cropHeight);\n    left = Math.min(left, imgLeft + imgWidth - cropWidth);\n\n    this.setStylesElements(this.cropArea(), { top, left }, true);\n  }\n\n  private zoomImage(scale: number) {\n    const [currWidth, currHeight, currTop, currLeft] = getStylesOfElement<number>(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);\n    const width = Math.max(currWidth * scale, 50);\n    const height = this.imageOrigin().nativeElement.offsetHeight * width / this.imageOrigin().nativeElement.offsetWidth;\n\n    // Calculate the center point of the current view\n    const centerX = currLeft + currWidth / 2;\n    const centerY = currTop + currHeight / 2;\n\n    // Calculate new position to maintain center point\n    const top = centerY - height / 2;\n    const left = centerX - width / 2;\n\n    // Get current crop area dimensions and position\n    let [cropWidth, cropHeight] = getStylesOfElement<number>(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight']);\n    const [cropTop, cropLeft] = getStylesOfElement<number>(this.cropArea().nativeElement, ['style.top', 'style.left']);\n\n    // Scale crop area dimensions proportionally\n    cropWidth = cropWidth * scale;\n    cropHeight = cropHeight * scale;\n\n    // Calculate new crop area position relative to the image\n    const cropTopNew = top + (cropTop - currTop) * scale;\n    const cropLeftNew = left + (cropLeft - currLeft) * scale;\n\n    // Ensure crop area stays within bounds\n    const maxCropWidth = width;\n    const maxCropHeight = height;\n    const minWidth = 20;\n    const minHeight = 20;\n\n    cropWidth = Math.min(cropWidth, maxCropWidth);\n    cropHeight = Math.min(cropHeight, maxCropHeight);\n    cropWidth = Math.max(cropWidth, minWidth);\n    cropHeight = Math.max(cropHeight, minHeight);\n\n    this.ratioValue.set(this.getCropRatioValue());\n    if (this.ratioValue()) {\n      cropHeight = cropWidth / this.ratioValue();\n      cropHeight = Math.min(cropHeight, maxCropHeight);\n      cropHeight = Math.max(cropHeight, minHeight);\n      cropWidth = cropHeight * this.ratioValue();\n    }\n\n    // Update clipping rectangle\n    this.rectClip.update(rect => {\n      const rectTop = cropTopNew - top;\n      const rectLeft = cropLeftNew - left;\n      return {\n        ...rect,\n        top: rectTop,\n        left: rectLeft,\n        right: rectLeft + cropWidth,\n        bottom: rectTop + cropHeight\n      };\n    });\n\n    const clip = `rect(${this.rectClip().top}px, ${this.rectClip().right}px, ${this.rectClip().bottom}px, ${this.rectClip().left}px)`;\n\n    // Update all elements with new positions and dimensions\n    this.setStylesElements(this.cropArea(), { top: cropTopNew, left: cropLeftNew, width: cropWidth, height: cropHeight });\n    this.setStylesElements(this.imageOrigin(), { top, left, width, height });\n    this.setCirclePosition({ top, left, width, height });\n    this.setStylesElements(this.imageClip(), { clip, top, left, width, height });\n    this.getRectClipImage();\n  }\n\n  protected handlerRotateImage(event: Event) {\n    event.stopPropagation();\n    this.image.src = this.imgSrc();\n    this.image.onload = () => {\n      const canvas = document.createElement('canvas');\n      const ctx = canvas.getContext(\"2d\");\n\n      const [width, height] = getStylesOfElement<number>(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight']);\n      const originWidth = this.image.width;\n      const originHeight = this.image.height;\n      let top = parseInt(this.imageOrigin().nativeElement.style.top);\n      let left = parseInt(this.imageOrigin().nativeElement.style.left);\n      let newHeight = width;\n      let newWidth = height;\n      const ratio = newWidth / newHeight;\n\n      canvas.height = originWidth;\n      canvas.width = originHeight;\n      ctx?.translate(originHeight / 2, originWidth / 2);\n      ctx?.rotate(Math.PI / 2);\n      ctx?.drawImage(this.image, -originWidth / 2, -originHeight / 2);\n      const src = getDataUrl(canvas, this.mimetype(), this.image.src);\n\n      this.imgSrc.set(src);\n      newWidth = Math.max(newWidth, originHeight);\n      newWidth = Math.min(newWidth, this.containerWidth());\n      newHeight = newWidth / ratio;\n      newHeight = Math.max(newHeight, originWidth);\n      newHeight = Math.min(newHeight, this.containerHeight());\n      newWidth = newHeight * ratio;\n      top = top - (newHeight / 2 - height / 2);\n      left = left - (newWidth / 2 - width / 2);\n      this.setStylesElements(this.imageOrigin(), { src, top, left, width: newWidth, height: newHeight });\n      this.setCirclePosition({ top, left, width: newWidth, height: newHeight });\n      this.setStylesElements(this.imageClip(), { src, top, left, width: newWidth, height: newHeight });\n      this.getOriginalImageSize(() => {\n        this.cropSize.set({ width: this.originWidth(), height: this.originHeight() });\n        this.updateOriginImageSize();\n      });\n      this.changedImage(true);\n    };\n  }\n\n  protected getOriginalImageSize(callback?: () => void) {\n    this.image.src = this.imgSrc();\n    this.image.onload = () => {\n      this.originWidth.set(this.image.width);\n      this.originHeight.set(this.image.height);\n      if (callback) {\n        return callback();\n      }\n    };\n  }\n\n  protected changedImage(status: boolean) {\n    this.changed.set(status);\n  }\n\n  protected handlerFlipImage(event: Event, mode: 'horizontal' | 'vertical') {\n    event.stopPropagation();\n    this.image.src = this.imgSrc();\n    this.image.onload = () => {\n      const canvas = document.createElement('canvas');\n      const ctx = canvas.getContext(\"2d\");\n      const width = this.image.width;\n      const height = this.image.height;\n\n      canvas.height = height;\n      canvas.width = width;\n      if (ctx) {\n        ctx.drawImage(this.image, 0, 0, width, height);\n        if (mode === 'horizontal') {\n          ctx.translate(width, 0);\n          ctx.scale(-1, 1);\n        }\n        if (mode === 'vertical') {\n          ctx.translate(0, height);\n          ctx.scale(1, -1);\n        }\n        ctx.clearRect(0, 0, width, height);\n        ctx.drawImage(this.image, 0, 0);\n      }\n      const dataUrl = getDataUrl(canvas, this.mimetype(), this.image.src);\n\n      this.imgSrc.set(dataUrl);\n      this.imageOrigin().nativeElement.src = dataUrl;\n      this.imageClip().nativeElement.src = dataUrl;\n      this.updateCropAreaPos();\n      this.updateRectClipPos();\n      this.changedImage(true);\n    };\n  }\n\n  protected handlerRestoreImage(event: Event) {\n    event.stopPropagation();\n    this.imgSrc.set(this.dataUrlOrigin());\n    this.dragGridCrop.set(false);\n    this.getOriginalImageSize(() => {\n      this.updateOriginImageSize();\n      this.changedImage(false);\n      this.cropRatioItemSelected.set('');\n      this.updateCropAreaPos();\n      this.updateRectClipPos();\n    });\n  }\n\n  private cropImage(): Promise<string> {\n    return new Promise((resolve) => {\n      this.image.src = this.imgSrc();\n      this.image.onload = () => {\n        const canvas = document.createElement('canvas');\n        const originWidth = this.image.width;\n        const width = parseInt(this.imageClip().nativeElement.offsetWidth);\n        const height = parseInt(this.imageClip().nativeElement.offsetHeight);\n        const scale = originWidth / width;\n        const rectClip = this.getRectClipImage();\n        const cropRect = getCropRectImage(rectClip, width, height, scale);\n\n        canvas.height = cropRect.height;\n        canvas.width = cropRect.width;\n        this.originHeight.set(canvas.height);\n        this.originWidth.set(canvas.width);\n        const ctx = canvas.getContext(\"2d\");\n        if (ctx) {\n          ctx.drawImage(this.image, cropRect.left, cropRect.top, cropRect.width, cropRect.height, 0, 0, cropRect.width, cropRect.height);\n        }\n        const dataUrl = getDataUrl(canvas, this.mimetype(), this.image.src);\n\n        this.changedImage(true);\n        resolve(dataUrl);\n      };\n    });\n  }\n\n  protected async handlerSaveFile(event: Event, modeSave: TYPE_MODE_SAVE) {\n    event.stopPropagation();\n    const dataUrl = await this.cropImage();\n    const file = convertBase64ToBlob(dataUrl);\n\n    this.outSaveFile.emit({ file: file, url: dataUrl, mode: modeSave });\n  }\n\n\n  protected async handlerClose(event: Event): Promise<void> {\n    event.stopPropagation();\n    this.outClose.emit({ isClickButtonClose: true });\n  }\n\n  protected handlerSelectCropRatioItem(event: Event, key: string) {\n    event.stopPropagation();\n    this.cropRatioItemSelected.set(key);\n    this.changed.set(true);\n    this.dragGridCrop.set(true);\n    this.updateOriginImageSize();\n  }\n\n  protected handlerResize(event: Event) {\n    event.stopPropagation();\n    if (this.resizeComponentRef) {\n      return;\n    }\n    this.resizeComponentRef = this.dynamicComponentService.resolveComponentFactory(LibsUiComponentsImageEditorResizeComponent);\n    const instance = this.resizeComponentRef.instance;\n    const subs = instance.outClose.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(() => {\n      this.dynamicComponentService.remove(this.resizeComponentRef);\n      this.resizeComponentRef = undefined;\n      subs.unsubscribe();\n    });\n\n    instance.src = this.imgSrc();\n    instance.mimetype = this.mimetype();\n    instance.zIndex = this.zIndex();\n    subs.add(instance.outSave.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(event => {\n      this.resizeData.set(event);\n      this.saveResize();\n    }));\n    instance.resizeData = { ratio: 100, width: this.originWidth(), height: this.originHeight() };\n    instance.originWidth = this.originWidth();\n    instance.originHeight = this.originHeight();\n    this.dynamicComponentService.addToBody(this.resizeComponentRef);\n  }\n\n  protected saveResize() {\n    this.image.src = this.imgSrc();\n    this.image.onload = () => {\n      const canvas = document.createElement('canvas');\n\n      canvas.width = this.resizeData().width;\n      canvas.height = this.resizeData().height;\n      const ctx = canvas.getContext('2d');\n\n      if (ctx) {\n        ctx.drawImage(this.image, 0, 0, canvas.width, canvas.height);\n      }\n      const src = getDataUrl(canvas, this.mimetype(), this.image.src);\n      const [currWidth, currHeight, currTop, currLeft] = getStylesOfElement<number>(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);\n      const ratio = currWidth / currHeight;\n      let newWidth = Math.min(this.resizeData().width, this.containerWidth());\n      let newHeight = newWidth / ratio;\n\n      newHeight = Math.min(this.resizeData().height, this.containerHeight());\n      newWidth = newHeight * ratio;\n      Math.min(this.resizeData().height, this.containerHeight());\n      const top = currTop + (currHeight - newHeight) / 2;\n      const left = currLeft + (currWidth - newWidth) / 2;\n      const movementY = currTop - top;\n      const movementX = currLeft - left;\n\n      this.rectClip.update(rect => {\n        const rectTop = rect.top + movementY;\n        const rectLeft = rect.left + movementX;\n        return {\n          ...rect,\n          top: rectTop,\n          left: rectLeft,\n          right: rectLeft + newWidth,\n          bottom: rectTop + newHeight\n        };\n      });\n      const clip = `rect(${this.rectClip().top}px, ${this.rectClip().right}px, ${this.rectClip().bottom}px, ${this.rectClip().left}px)`;\n\n      this.setStylesElements(this.imageOrigin(), { src, top, left, width: newWidth, height: newHeight });\n      this.setCirclePosition({ top, left, width: newWidth, height: newHeight });\n      this.setStylesElements(this.imageClip(), { src, clip, top, left, width: newWidth, height: newHeight });\n      this.imgSrc.set(src);\n      this.originWidth.set(this.resizeData().width);\n      this.originHeight.set(this.resizeData().height);\n      this.updateCropAreaPos();\n      this.updateRectClipPos();\n      this.changedImage(true);\n    };\n  }\n\n  protected handlerCropWidth() {\n    this.processCropWidthOrHeight('width');\n  }\n\n  protected handlerCropHeight() {\n    this.processCropWidthOrHeight('height');\n  }\n\n  private processCropWidthOrHeight(cropBy: 'width' | 'height') {\n    this.ratioValue.set(this.getCropRatioValue());\n    const [currImgWidth, currImgHeight, currImgTop, currImgLeft] = getStylesOfElement<number>(this.imageOrigin().nativeElement, ['offsetWidth', 'offsetHeight', 'style.top', 'style.left']);\n    let [cropWidth, cropHeight] = getStylesOfElement<number>(this.cropArea().nativeElement, ['offsetWidth', 'offsetHeight']);\n    const [cropTop, cropLeft] = getStylesOfElement<number>(this.cropArea().nativeElement, ['style.top', 'style.left']);\n    const maxCropWidth = currImgLeft + currImgWidth - cropLeft;\n    const maxCropHeight = currImgHeight + currImgTop - cropTop;\n    const ratio = this.originWidth() / currImgWidth;\n    let cropValue = get(this.cropSize, cropBy) / ratio;\n\n    cropValue = Math.min(cropValue, cropBy === 'width' ? maxCropWidth : maxCropHeight);\n    cropValue = Math.max(1, cropValue);\n    if (this.ratioValue()) {\n      if (cropBy === 'width') {\n        cropHeight = cropValue / this.ratioValue();\n        cropHeight = Math.min(cropHeight, maxCropHeight);\n        cropValue = cropHeight * this.ratioValue();\n        this.cropSize.update(cropSize => ({ ...cropSize, height: Math.round(cropHeight * ratio) }));\n        this.cropArea().nativeElement.style.height = `${cropHeight}px`;\n      }\n      if (cropBy === 'height') {\n        cropWidth = cropValue * this.ratioValue();\n        cropWidth = Math.min(cropWidth, maxCropWidth);\n        cropValue = cropWidth / this.ratioValue();\n        this.cropSize.update(cropSize => ({ ...cropSize, width: Math.round(cropWidth * ratio) }));\n        this.cropArea().nativeElement.style.width = `${cropWidth}px`;\n      }\n    }\n    this.cropSize.update(cropSize => ({ ...cropSize, [cropBy]: Math.round(cropValue * ratio) }));\n    this.cropArea().nativeElement.style[cropBy] = `${cropValue}px`;\n    const imageRect = this.imageClip().nativeElement.getBoundingClientRect();\n    const cropRect = this.cropArea().nativeElement.getBoundingClientRect();\n    const { width, height } = cropRect;\n    const left = cropRect.left - imageRect.left;\n    const top = cropRect.top - imageRect.top;\n    const right = left + width;\n    const bottom = top + height;\n\n    this.imageClip().nativeElement.style.clip = `rect(${top}px, ${right}px, ${bottom}px, ${left}px)`;\n  }\n\n  private setStylesElements(element: ElementRef, styles: IStyles, isUpdateRectClipPos?: boolean) {\n    Object.keys(styles).forEach(key => {\n      const value = get(styles, key as keyof IStyles);\n      if (isNil(value)) {\n        return;\n      }\n      set(element.nativeElement.style, key, typeof value === 'number' ? `${value}px` : value);\n    });\n    if (isUpdateRectClipPos) {\n      this.updateRectClipPos();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.dynamicComponentService.remove(this.resizeComponentRef);\n  }\n}\n","<libs_ui-components-modal [width]=\"'calc(100vw - 400px)'\"\n  [height]=\"'calc(100vh - 100px)'\"\n  [minWidth]=\"'1200px'\"\n  [headerConfig]=\"{ignoreHeaderTheme: true, removeButtonClose: true}\"\n  [bodyConfig]=\"{classInclude: '!p-0', scrollOverlayOptions: {scrollX: 'hidden', scrollY: 'hidden'}}\"\n  [footerConfig]=\"{hidden: true}\"\n  [zIndex]=\"zIndex()\"\n  [buttonsFooter]=\"[]\"\n  [mode]=\"'center'\">\n  <div class=\"libs-ui-modal-header-custom w-full flex items-center justify-between\">\n    <div class=\"w-full relative mx-[24px] h-[16px]\">\n      <div class=\"flex w-full libs-ui-font-h4s absolute\">\n        <libs_ui-components-popover [type]=\"'text'\"\n          [config]=\"{zIndex: zIndex()}\">\n          {{ 'i18n_edit_file_name' | translate:{file_name: nameFile()} }}\n        </libs_ui-components-popover>\n      </div>\n    </div>\n    <div class=\"flex items-center\">\n      @if (modeShowButton() === 'save-file') {\n        <libs_ui-components-buttons-button [type]=\"'button-third'\"\n          [label]=\"'i18n_cancel'\"\n          [disable]=\"loading() || loadingImage()\"\n          [classInclude]=\"'py-[6px] px-[12px] mr-[16px]'\"\n          (outClick)=\"handlerClose($event)\" />\n\n        <libs_ui-components-buttons-button class=\"mr-[24px]\"\n          [label]=\"'i18n_save'\"\n          [disable]=\"loading() || loadingImage()\"\n          [classInclude]=\"'py-[6px] px-[12px]'\"\n          (outClick)=\"handlerSaveFile($event,'save-file')\" />\n      }\n      @else {\n        <libs_ui-components-buttons-button [label]=\"'i18n_save_to_new_file'\"\n          [type]=\"'button-outline'\"\n          [classInclude]=\"'mx-[16px] py-[6px] px-[12px]'\"\n          [disable]=\"loading() || loadingImage() || !changed()\"\n          (outClick)=\"handlerSaveFile($event,'save-api-as-new-file')\" />\n        <libs_ui-components-buttons-button [type]=\"'button-secondary'\"\n          [label]=\"'i18n_save'\"\n          [classInclude]=\"'py-[6px] px-[12px]'\"\n          [disable]=\"loading() || loadingImage() || !changed()\"\n          (outClick)=\"handlerSaveFile($event,'save-api')\" />\n        <div class=\"h-[12px] libs-ui-border-left-general mx-[16px]\"></div>\n        <libs_ui-components-buttons-button class=\"mr-[24px]\"\n          [classInclude]=\"'p-[6px]'\"\n          [type]=\"'button-third-hover-danger'\"\n          [iconOnlyType]=\"true\"\n          [disable]=\"loading()\"\n          [popover]=\"{config: {content: 'i18n_exit_edit_mode', zIndex: zIndex()}}\"\n          [classIconLeft]=\"'libs-ui-icon-close'\"\n          (outClick)=\"handlerClose($event)\" />\n      }\n    </div>\n  </div>\n  <div class=\"libs-ui-modal-body-custom h-full w-full flex\">\n    <div #imageEditorContainer\n      class=\"relative w-full h-full bg-[#e6e7ea] border-radius-bottom-left-8px p-[32px] libs-ui-input-image-editor-container-left\">\n      @if (loadingImage()) {\n        <div class=\"absolute h-full w-full bg-[#e6e7ea] top-0 left-0 z-[5]\">\n          <libs_ui-components-spinner [size]=\"'medium'\" />\n        </div>\n      }\n      <div class=\"absolute h-[32px] w-[calc(100%-64px)] bg-[#e6e7eab8] top-0 left-[32px] z-[1]\">\n      </div>\n      <div class=\"absolute w-[32px] h-full bg-[#e6e7eab8] left-0 top-0 z-[1] rounded-bl-[8px]\">\n      </div>\n      <div class=\"absolute w-[32px] h-full bg-[#e6e7eab8] right-0 top-0 z-[1]\">\n      </div>\n      <div class=\"absolute h-[32px] w-[calc(100%-64px)] bg-[#e6e7eab8] bottom-0 left-[32px] z-[1]\">\n      </div>\n      <div #imageContainer\n        class=\"relative w-full h-full rounded-bl-[8px] min-h-0\">\n        <div #circleTL\n          class=\"absolute z-[1]\"\n          [class.hidden]=\"!dragGridCrop()\"\n          [class.flex]=\"dragGridCrop()\">\n          <ng-template *ngTemplateOutlet=\"svgCircle\" />\n        </div>\n        <div #circleTR\n          class=\"absolute z-[1]\"\n          [class.hidden]=\"!dragGridCrop()\"\n          [class.flex]=\"dragGridCrop()\">\n          <ng-template *ngTemplateOutlet=\"svgCircle\" />\n        </div>\n        <div #circleBL\n          class=\"absolute z-[1]\"\n          [class.hidden]=\"!dragGridCrop()\"\n          [class.flex]=\"dragGridCrop()\">\n          <ng-template *ngTemplateOutlet=\"svgCircle\" />\n        </div>\n        <div #circleBR\n          class=\"absolute z-[1]\"\n          [class.hidden]=\"!dragGridCrop()\"\n          [class.flex]=\"dragGridCrop()\">\n          <ng-template *ngTemplateOutlet=\"svgCircle\" />\n        </div>\n        <img #imageOrigin\n          class=\"absolute border-[4px] border-[#8e61ee] rounded-[4px] top-[-1000px] left-[-1000px]\"\n          loading=\"lazy\"\n          [src]=\"imgSrc()\"\n          (load)=\"handlerImageLoaded()\"\n          (error)=\"handlerImageError($event)\" />\n        <img #imageClip\n          class=\"absolute z-[1]  top-[-1000px] left-[-1000px]\"\n          loading=\"lazy\"\n          [src]=\"imgSrc()\"\n          (error)=\"handlerImageError($event)\" />\n        <div class=\"absolute top-0 left-0 right-0 bottom-0 bg-[#e6e7eab8]\">\n        </div>\n        <div #cropArea\n          class=\"absolute z-[2] border-[4px] border-[#8e61ee] rounded-[4px]\">\n          <div class=\"relative w-full h-full\">\n            <div #cropTL\n              class=\"absolute z-[2] left-[-18px] top-[-17px] cursor-nwse-resize\">\n              <ng-template *ngTemplateOutlet=\"svgArrow\" />\n            </div>\n            <div #cropBL\n              class=\"absolute z-[2] left-[-17px] bottom-[-18px] cursor-nesw-resize rotate-[270deg]\">\n              <ng-template *ngTemplateOutlet=\"svgArrow\" />\n            </div>\n            <div #cropTR\n              class=\"absolute z-[2] right-[-17px] top-[-17px] cursor-nesw-resize rotate-[90deg]\">\n              <ng-template *ngTemplateOutlet=\"svgArrow\" />\n            </div>\n            <div #cropBR\n              class=\"absolute z-[2] right-[-18px] bottom-[-17px] cursor-nwse-resize rotate-[180deg]\">\n              <ng-template *ngTemplateOutlet=\"svgArrow\" />\n            </div>\n            <div #cropLineVL\n              class=\"absolute z-[1] w-[12px] left-[-6px] top-0 bottom-0 cursor-ew-resize\">\n            </div>\n            <div #cropLineVR\n              class=\"absolute z-[1] w-[12px] right-[-6px] top-0 bottom-0 cursor-ew-resize\">\n            </div>\n            <div #cropLineHT\n              class=\"absolute z-[1] h-[12px] left-0 top-[-6px] right-0 cursor-ns-resize\">\n            </div>\n            <div #cropLineHB\n              class=\"absolute z-[1] h-[12px] left-0 right-0 bottom-[-6px] cursor-ns-resize\">\n            </div>\n\n            <div class=\"bg-white absolute h-[1px] top-0 left-0 right-0\"></div>\n            <div class=\"bg-white absolute h-[1px] bottom-0 left-0 right-0\"></div>\n            <div class=\"bg-white absolute w-[1px] top-0 left-0 bottom-0\"></div>\n            <div class=\"bg-white absolute w-[1px] top-0 bottom-0 right-0\"></div>\n            <div class=\"bg-white absolute h-[2px] top-[33%] left-0 right-0\"></div>\n            <div class=\"bg-white absolute h-[2px] top-[66%] left-0 right-0\"></div>\n            <div class=\"bg-white absolute w-[2px] top-0 bottom-0 left-[33%]\"></div>\n            <div class=\"bg-white absolute w-[2px] top-0 bottom-0 left-[66%]\"></div>\n          </div>\n        </div>\n      </div>\n      @if (loading()) {\n        <libs_ui-components-spinner [size]=\"'medium'\" />\n      }\n    </div>\n    <div class=\"w-[320px] shrink-0 p-[16px] bg-[#ffffff] rounded-br-[8px] z-[2] libs-ui-input-image-editor-container-right\">\n      <div class=\"flex items-center justify-between mb-[16px]\">\n        <div class=\"libs-ui-font-h4m\">{{ 'i18n_cutting_ratio' | translate }}</div>\n        <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n          [type]=\"'button-third'\"\n          [classIconLeft]=\"'libs-ui-icon-refresh'\"\n          [classInclude]=\"'p-[6px] mr-[8px]'\"\n          [popover]=\"{config: {content: 'i18n_restore_the_original_state', zIndex: zIndex()}}\"\n          [disable]=\"loading() || loadingImage() || !changed()\"\n          (outClick)=\"handlerRestoreImage($event)\" />\n      </div>\n      <div class=\"mb-[16px] libs-ui-image-editor-edit-display-grid-gap-16\">\n        @for (item of cropRatioItems(); track item.key) {\n          <div class=\"flex flex-col items-center\">\n            <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n              [sizeButton]=\"'large'\"\n              [disable]=\"loading() || loadingImage() || requiredCropFollowRatio()\"\n              [type]=\"'button-outline'\"\n              [classIconLeft]=\"item.icon+' !text-[20px]'\"\n              [isActive]=\"item.key === cropRatioItemSelected()\"\n              (outClick)=\"handlerSelectCropRatioItem($event, item.key)\" />\n            <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n              {{ (item.key === 'free' ? 'i18n_custom' : item.key) | translate }}\n            </div>\n          </div>\n        }\n      </div>\n      <div class=\"mb-[16px] flex\">\n        <libs_ui-components-inputs-valid class=\"mr-[16px]\"\n          [dataType]=\"'int'\"\n          [labelConfig]=\"{labelLeft: 'i18n_width', classInclude: 'mb-[4px]', labelLeftClass: 'libs-ui-font-h7r'}\"\n          [fieldNameBind]=\"'width'\"\n          [(item)]=\"cropSize\"\n          [unitsRight]=\"[{id: 'px', label: 'px'}]\"\n          [keySelectedUnitRight]=\"'px'\"\n          [valueUpDownNumber]=\"1\"\n          [maxValueNumber]=\"originWidth()\"\n          [disable]=\"loading() || loadingImage()\"\n          (outValueChange)=\"handlerCropWidth()\" />\n        <libs_ui-components-inputs-valid [dataType]=\"'int'\"\n          [labelConfig]=\"{labelLeft: 'i18n_length', classInclude: 'mb-[4px]', labelLeftClass: 'libs-ui-font-h7r'}\"\n          [fieldNameBind]=\"'height'\"\n          [(item)]=\"cropSize\"\n          [unitsRight]=\"[{id: 'px', label: 'px'}]\"\n          [keySelectedUnitRight]=\"'px'\"\n          [valueUpDownNumber]=\"1\"\n          [maxValueNumber]=\"originHeight()\"\n          [disable]=\"loading() || loadingImage()\"\n          (outValueChange)=\"handlerCropHeight()\" />\n      </div>\n      <div class=\"libs-ui-font-h4m mb-[16px]\">{{ 'i18n_rotate_photos' | translate }}</div>\n      <div class=\"flex libs-ui-image-editor-edit-display-grid-gap-16\">\n        <div class=\"flex flex-col items-center\">\n          <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n            [sizeButton]=\"'large'\"\n            [disable]=\"loading() || loadingImage()\"\n            [type]=\"'button-outline'\"\n            [classIconLeft]=\"'libs-ui-icon-rotate-image-outline !text-[20px]'\"\n            (outClick)=\"handlerRotateImage($event)\" />\n          <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n            {{ 'i18n_rotate_ninety_degrees' | translate }}\n          </div>\n        </div>\n\n        <div class=\"flex flex-col items-center\">\n          <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n            [sizeButton]=\"'large'\"\n            [disable]=\"loading() || loadingImage()\"\n            [type]=\"'button-outline'\"\n            [classIconLeft]=\"'libs-ui-icon-flip-vertical !text-[20px]'\"\n            (outClick)=\"handlerFlipImage($event, 'vertical')\" />\n          <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n            {{ 'i18n_vertical_flip' | translate }}\n          </div>\n        </div>\n\n        <div class=\"flex flex-col items-center\">\n          <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n            [sizeButton]=\"'large'\"\n            [disable]=\"loading() || loadingImage()\"\n            [type]=\"'button-outline'\"\n            [classIconLeft]=\"'libs-ui-icon-flip-horizontal !text-[20px]'\"\n            (outClick)=\"handlerFlipImage($event, 'horizontal')\" />\n          <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n            {{ 'i18n_horizontal_flip' | translate }}\n          </div>\n        </div>\n\n        <div class=\"flex flex-col items-center\">\n          <libs_ui-components-buttons-button [iconOnlyType]=\"true\"\n            [sizeButton]=\"'large'\"\n            [disable]=\"loading() || loadingImage()\"\n            [type]=\"'button-outline'\"\n            [classIconLeft]=\"'libs-ui-icon-scale !text-[20px]'\"\n            (outClick)=\"handlerResize($event)\" />\n          <div class=\"libs-ui-font-h6r text-[#6a7383] mt-[4px]\">\n            {{ 'i18n_resize' | translate }}\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n</libs_ui-components-modal>\n\n<ng-template #svgArrow>\n  <svg xmlns=\"http://www.w3.org/2000/svg\"\n    width=\"40\"\n    height=\"40\"\n    viewBox=\"0 0 40 40\"\n    fill=\"none\">\n    <path d=\"M33.6987 14.6969L18.3309 14.6969C16.516 14.6969 15.0311 16.1818 15.0311 17.9967L15.0311 33.3645\"\n      stroke=\"white\"\n      stroke-width=\"5.83333\"\n      stroke-miterlimit=\"10\"\n      stroke-linecap=\"round\"\n      stroke-linejoin=\"round\" />\n  </svg>\n</ng-template>\n\n<ng-template #svgCircle>\n  <svg xmlns=\"http://www.w3.org/2000/svg\"\n    width=\"15\"\n    height=\"15\"\n    viewBox=\"0 0 15 15\"\n    fill=\"none\">\n    <circle cx=\"7.5\"\n      cy=\"7.5\"\n      r=\"7.5\"\n      fill=\"white\" />\n  </svg>\n</ng-template>\n"]}