keevo-components 1.8.340 → 1.8.341
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/api/services/image.cutter.service.mjs +564 -0
- package/esm2022/lib/components/kv-avatar/kv-avatar.component.mjs +54 -231
- package/esm2022/lib/components/kv-avatar/kv-avatar.module.mjs +7 -3
- package/esm2022/lib/components/kv-home-card/kv-home-card.module.mjs +1 -1
- package/esm2022/lib/components/kv-image-upload/kv-image-upload.component.mjs +46 -16
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/keevo-components.mjs +816 -413
- package/fesm2022/keevo-components.mjs.map +1 -1
- package/lib/api/services/image.cutter.service.d.ts +15 -0
- package/lib/components/kv-avatar/kv-avatar.component.d.ts +12 -37
- package/lib/components/kv-avatar/kv-avatar.module.d.ts +2 -1
- package/lib/components/kv-image-upload/kv-image-upload.component.d.ts +10 -3
- package/lib/components/kv-table/kv-table.component.d.ts +1 -1
- package/lib/components/kv-table-expandable/kv-table-expandable.component.d.ts +1 -1
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,564 @@
|
|
|
1
|
+
import { Component, computed, HostListener, Injectable, signal, ViewChild } from '@angular/core';
|
|
2
|
+
import { DialogModule } from 'primeng/dialog';
|
|
3
|
+
// ? Componente de modal usado no cutter
|
|
4
|
+
import { SliderModule } from 'primeng/slider';
|
|
5
|
+
import { CommonModule } from '@angular/common';
|
|
6
|
+
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
7
|
+
import { ButtonModule } from 'primeng/button';
|
|
8
|
+
import { FileUploadModule } from 'primeng/fileupload';
|
|
9
|
+
import { KvLabelModule } from '../../components/kv-label/kv-label.module';
|
|
10
|
+
import { FormService } from './form.service';
|
|
11
|
+
import { KvModalModule } from '../../components/kv-modal/kv-modal.module';
|
|
12
|
+
import { KvButtonModule } from '../../components/kv-button/kv-button.module';
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
import * as i1 from "primeng/dynamicdialog";
|
|
15
|
+
import * as i2 from "primeng/slider";
|
|
16
|
+
import * as i3 from "@angular/forms";
|
|
17
|
+
import * as i4 from "../../components/kv-button/kv-button.component";
|
|
18
|
+
import * as i5 from "../../components/kv-label/kv-label.component";
|
|
19
|
+
export class ImageCutterService {
|
|
20
|
+
constructor(dialogservice) {
|
|
21
|
+
this.dialogservice = dialogservice;
|
|
22
|
+
}
|
|
23
|
+
cut(imageBase64, width, height, indCircle = false) {
|
|
24
|
+
const cutConfig = { imageBase64, width, height, indCircle };
|
|
25
|
+
this.abrirModal(cutConfig);
|
|
26
|
+
return new Promise((resolve, reject) => {
|
|
27
|
+
window.addEventListener('cropImage', (event) => {
|
|
28
|
+
if (event.detail.status == true) {
|
|
29
|
+
resolve(event.detail.result);
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
reject('Operacao cancelada.');
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
abrirModal(cutConfig) {
|
|
38
|
+
const params = {
|
|
39
|
+
closable: true,
|
|
40
|
+
maximizable: false,
|
|
41
|
+
popup: true,
|
|
42
|
+
width: '75vw',
|
|
43
|
+
header: `Cortar`,
|
|
44
|
+
id: { imagem: cutConfig.imageBase64, width: cutConfig.width, height: cutConfig.height, indCircle: cutConfig.indCircle },
|
|
45
|
+
styleClass: 'modal'
|
|
46
|
+
};
|
|
47
|
+
const dialogRef = FormService.openDialog(this.dialogservice, SimpleComponent, () => { }, params);
|
|
48
|
+
}
|
|
49
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: ImageCutterService, deps: [{ token: i1.DialogService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
50
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: ImageCutterService, providedIn: 'root' }); }
|
|
51
|
+
}
|
|
52
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: ImageCutterService, decorators: [{
|
|
53
|
+
type: Injectable,
|
|
54
|
+
args: [{
|
|
55
|
+
providedIn: 'root'
|
|
56
|
+
}]
|
|
57
|
+
}], ctorParameters: () => [{ type: i1.DialogService }] });
|
|
58
|
+
class SimpleComponent {
|
|
59
|
+
constructor(dynamicDialogRef, dialogService) {
|
|
60
|
+
this.dynamicDialogRef = dynamicDialogRef;
|
|
61
|
+
this.dialogService = dialogService;
|
|
62
|
+
// ? Variavies que recebem os valores do dialog
|
|
63
|
+
this._variaviesDialog = signal(null);
|
|
64
|
+
//? Referências para o contexto e canvas
|
|
65
|
+
this.canvas = signal(null);
|
|
66
|
+
this.ctx = signal(null);
|
|
67
|
+
this._Image = signal(new Image());
|
|
68
|
+
//? Coordenadas iniciais e finais do desenho
|
|
69
|
+
this.startX = 0;
|
|
70
|
+
this.startY = 0;
|
|
71
|
+
//? Fator de escala da imagem e zoom
|
|
72
|
+
this.scaleFactorReadOnly = computed(() => {
|
|
73
|
+
return Math.min(this.scaleFactor(), 3);
|
|
74
|
+
});
|
|
75
|
+
this.scaleFactor = signal(null);
|
|
76
|
+
this.scaleFactorModel = 0;
|
|
77
|
+
this.imageX = 0;
|
|
78
|
+
this.imageY = 0;
|
|
79
|
+
//#region Propriedades computadas de tamanho do canvas e da imagem
|
|
80
|
+
//? Largura e altura do canvas baseado na tamanho da tela
|
|
81
|
+
this.canvasWidth = computed(() => this.widthTela() * 0.5);
|
|
82
|
+
this.canvasHeight = computed(() => this.heigthTela() * 0.5);
|
|
83
|
+
//? Largura e altura da área de corte
|
|
84
|
+
this.cutWidth = computed(() => {
|
|
85
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
86
|
+
return this.radius() * 2;
|
|
87
|
+
}
|
|
88
|
+
else {
|
|
89
|
+
const canvasWidth = this.canvasWidth();
|
|
90
|
+
const maxWidth = canvasWidth * 0.8;
|
|
91
|
+
const proportionalHeight = maxWidth * this.proporcaoImagem();
|
|
92
|
+
return proportionalHeight > this.canvasHeight() * 0.8 ? this.canvasHeight() * 0.8 / this.proporcaoImagem() : maxWidth;
|
|
93
|
+
}
|
|
94
|
+
});
|
|
95
|
+
this.cutHeight = computed(() => {
|
|
96
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
97
|
+
return this.radius() * 2;
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
const canvasHeight = this.canvasHeight();
|
|
101
|
+
const maxHeight = canvasHeight * 0.8;
|
|
102
|
+
const proportionalWidth = maxHeight / this.proporcaoImagem();
|
|
103
|
+
return proportionalWidth > this.canvasWidth() * 0.8 ? this.canvasWidth() * 0.8 * this.proporcaoImagem() : maxHeight;
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
//? Largura e altura da imagem no canvas (baseado na escala) - tamanho real da imagem
|
|
107
|
+
this.canvaImageWidth = computed(() => {
|
|
108
|
+
return this._Image().width * this.scaleFactorReadOnly();
|
|
109
|
+
});
|
|
110
|
+
this.canvaImageHeight = computed(() => {
|
|
111
|
+
return this._Image().height * this.scaleFactorReadOnly();
|
|
112
|
+
});
|
|
113
|
+
this.proporcaoImagem = computed(() => {
|
|
114
|
+
return this._variaviesDialog()?.height / this._variaviesDialog()?.width;
|
|
115
|
+
});
|
|
116
|
+
this.radius = computed(() => {
|
|
117
|
+
const minSize = Math.min(this.canvasHeight() * 0.8, this.canvasWidth() * 0.8);
|
|
118
|
+
return minSize / 2;
|
|
119
|
+
});
|
|
120
|
+
//#endregion
|
|
121
|
+
// #region Limites de movimentação da imagem
|
|
122
|
+
//? Limites de movimentação da imagem
|
|
123
|
+
this.indCircle = computed(() => {
|
|
124
|
+
return this._variaviesDialog()?.indCircle;
|
|
125
|
+
});
|
|
126
|
+
this.xMin = computed(() => {
|
|
127
|
+
return !this._variaviesDialog()?.indCircle ? (this.canvasWidth() - this.cutWidth()) / 2 : (this.canvasWidth() - (this.radius() * 2)) / 2;
|
|
128
|
+
});
|
|
129
|
+
this.yMin = computed(() => {
|
|
130
|
+
return !this._variaviesDialog()?.indCircle ? (this.canvasHeight() - this.cutHeight()) / 2 : (this.canvasHeight() - (this.radius() * 2)) / 2;
|
|
131
|
+
});
|
|
132
|
+
this.xMax = computed(() => {
|
|
133
|
+
return !this._variaviesDialog()?.indCircle ? -(this.canvaImageWidth() - (this.xMin()) - this.cutWidth()) : -(this.canvaImageWidth() - (this.xMin()) - (this.radius() * 2));
|
|
134
|
+
});
|
|
135
|
+
this.yMax = computed(() => {
|
|
136
|
+
return !this._variaviesDialog()?.indCircle ? -(this.canvaImageHeight() - (this.yMin()) - this.cutHeight()) : -(this.canvaImageHeight() - (this.yMin()) - (this.radius() * 2));
|
|
137
|
+
});
|
|
138
|
+
// #endregion
|
|
139
|
+
//? Proporção da imagem
|
|
140
|
+
this.widthTela = signal(0);
|
|
141
|
+
this.heigthTela = signal(0);
|
|
142
|
+
this.atualizaTamanhoTela();
|
|
143
|
+
}
|
|
144
|
+
setScaleFactorModel(valor) {
|
|
145
|
+
this.scaleFactorModel = valor;
|
|
146
|
+
this.scaleFactor.set(this.scaleFactorModel);
|
|
147
|
+
}
|
|
148
|
+
onWindowResize() {
|
|
149
|
+
this.atualizaTamanhoTela();
|
|
150
|
+
this.updateCanvasSize();
|
|
151
|
+
this.atualizaCutBorder();
|
|
152
|
+
}
|
|
153
|
+
atualizaTamanhoTela() {
|
|
154
|
+
this.widthTela.set(window.innerWidth);
|
|
155
|
+
this.heigthTela.set(window.innerHeight);
|
|
156
|
+
}
|
|
157
|
+
ngAfterViewInit() {
|
|
158
|
+
this.recuperarVariaviesDialog();
|
|
159
|
+
this.inciarReferencias();
|
|
160
|
+
this.iniciarCanvas();
|
|
161
|
+
}
|
|
162
|
+
inciarReferencias() {
|
|
163
|
+
this.canvas.set(this.cropCanvas.nativeElement);
|
|
164
|
+
this.ctx.set(this.canvas().getContext('2d'));
|
|
165
|
+
this._Image().src = this._variaviesDialog()?.imagem;
|
|
166
|
+
}
|
|
167
|
+
iniciarCanvas() {
|
|
168
|
+
this._Image().onload = () => {
|
|
169
|
+
//? Define o tamanho do canvas
|
|
170
|
+
this.canvas().height = this.canvasHeight();
|
|
171
|
+
this.canvas().width = this.canvasWidth();
|
|
172
|
+
this.ajustarZommEPosicaoInicial();
|
|
173
|
+
this.initializeImageMove(this.canvas(), this.ctx(), this._Image());
|
|
174
|
+
this.initializeMouseWheel(this.canvas(), this.ctx(), this._Image());
|
|
175
|
+
this.redrawCanvas(this.ctx(), this._Image());
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
ajustarZommEPosicaoInicial() {
|
|
179
|
+
const cutWidth = this.cutWidth();
|
|
180
|
+
const cutHeight = this.cutHeight();
|
|
181
|
+
const imageWidth = this._Image().width;
|
|
182
|
+
const imageHeight = this._Image().height;
|
|
183
|
+
//? Ajusta possição e zoom inicial
|
|
184
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
185
|
+
const scaleFactor = (this.radius() * 2) / Math.min(imageWidth, imageHeight);
|
|
186
|
+
this.setScaleFactorModel(scaleFactor);
|
|
187
|
+
this.scaleFactor.set(scaleFactor);
|
|
188
|
+
}
|
|
189
|
+
else {
|
|
190
|
+
const scaleFactor = this.cutHeight() === this.cutWidth()
|
|
191
|
+
? this._Image().width > this._Image().height
|
|
192
|
+
? this.cutHeight() / imageHeight
|
|
193
|
+
: this.cutWidth() / imageWidth
|
|
194
|
+
: this.cutHeight() > this.cutWidth()
|
|
195
|
+
? this.cutHeight() / imageHeight
|
|
196
|
+
: this.cutWidth() / imageWidth;
|
|
197
|
+
this.setScaleFactorModel(scaleFactor);
|
|
198
|
+
this.scaleFactor.set(scaleFactor);
|
|
199
|
+
}
|
|
200
|
+
this.imageX = Math.max((this.canvas().width - imageWidth * this.scaleFactorReadOnly()) / 2, this.xMin());
|
|
201
|
+
this.imageY = Math.max((this.canvasHeight() - imageHeight * this.scaleFactorReadOnly()) / 2, this.yMin());
|
|
202
|
+
}
|
|
203
|
+
//#region Métodos de recuperação de dados
|
|
204
|
+
recuperaInstanciaDialog() {
|
|
205
|
+
const arr = this.dialogService.dialogComponentRefMap;
|
|
206
|
+
const entries = arr.entries();
|
|
207
|
+
const firstEntry = entries.next().value;
|
|
208
|
+
const [firstKey, firstValue] = firstEntry;
|
|
209
|
+
const instance = this.dialogService.getInstance(firstKey);
|
|
210
|
+
this.dialogInstance = instance;
|
|
211
|
+
return instance;
|
|
212
|
+
}
|
|
213
|
+
recuperarVariaviesDialog() {
|
|
214
|
+
this._variaviesDialog.set(this.recuperaInstanciaDialog().data.id);
|
|
215
|
+
}
|
|
216
|
+
//#endregion
|
|
217
|
+
//#region Métodos de atualizacao do canvas
|
|
218
|
+
//? Atualiza o tamanho do canvas
|
|
219
|
+
updateCanvasSize() {
|
|
220
|
+
this.imageX = (this.canvas().width - this._Image().width * this.scaleFactorReadOnly()) / 2;
|
|
221
|
+
this.imageY = (this.canvasHeight() / 2) - (this._Image().height * this.scaleFactorReadOnly()) / 2;
|
|
222
|
+
this.canvas().height = this.canvasHeight();
|
|
223
|
+
this.canvas().width = this.canvasWidth();
|
|
224
|
+
this.redrawCanvas(this.ctx(), this._Image());
|
|
225
|
+
}
|
|
226
|
+
redrawCanvas(ctx, img) {
|
|
227
|
+
if (!ctx)
|
|
228
|
+
return;
|
|
229
|
+
this.drawCropArea(ctx);
|
|
230
|
+
ctx.clearRect(0, 0, ctx.canvas.width, ctx.canvas.height);
|
|
231
|
+
ctx.filter = 'blur(5px) opacity(0.9)';
|
|
232
|
+
ctx.drawImage(img, this.imageX, this.imageY, img.width * this.scaleFactorReadOnly(), img.height * this.scaleFactorReadOnly());
|
|
233
|
+
ctx.filter = 'none';
|
|
234
|
+
this.drawImageInCropArea(ctx, img);
|
|
235
|
+
}
|
|
236
|
+
//? Desenha a área de corte
|
|
237
|
+
drawCropArea(ctx) {
|
|
238
|
+
if (!ctx)
|
|
239
|
+
return;
|
|
240
|
+
this.atualizaCutBorder();
|
|
241
|
+
const cropX = ctx.canvas.width / 2 - this.cutWidth() / 2;
|
|
242
|
+
const cropY = ctx.canvas.height / 2 - this.cutHeight() / 2;
|
|
243
|
+
const cropXCircle = ctx.canvas.width / 2 - this.radius() / 2;
|
|
244
|
+
const cropYCircle = ctx.canvas.height / 2 - this.radius() / 2;
|
|
245
|
+
ctx.beginPath();
|
|
246
|
+
ctx.strokeStyle = 'transparent';
|
|
247
|
+
ctx.lineWidth = 2;
|
|
248
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
249
|
+
ctx.arc(cropXCircle, cropYCircle, this.radius(), 0, Math.PI * 2, true);
|
|
250
|
+
}
|
|
251
|
+
else {
|
|
252
|
+
ctx.rect(cropX, cropY, this.cutWidth(), this.cutHeight());
|
|
253
|
+
}
|
|
254
|
+
ctx.stroke();
|
|
255
|
+
}
|
|
256
|
+
//? Desenha a imagem na área de corte
|
|
257
|
+
drawImageInCropArea(ctx, img) {
|
|
258
|
+
if (!ctx)
|
|
259
|
+
return;
|
|
260
|
+
const cropX = (ctx.canvas.width - this.cutWidth()) / 2;
|
|
261
|
+
const cropY = (ctx.canvas.height - this.cutHeight()) / 2;
|
|
262
|
+
ctx.save();
|
|
263
|
+
ctx.beginPath();
|
|
264
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
265
|
+
ctx.arc(ctx.canvas.width / 2, ctx.canvas.height / 2, this.radius(), 0, Math.PI * 2, true);
|
|
266
|
+
}
|
|
267
|
+
else {
|
|
268
|
+
ctx.rect(cropX, cropY, this.cutWidth(), this.cutHeight());
|
|
269
|
+
}
|
|
270
|
+
ctx.clip();
|
|
271
|
+
ctx.drawImage(img, this.imageX, this.imageY, img.width * this.scaleFactorReadOnly(), img.height * this.scaleFactorReadOnly());
|
|
272
|
+
ctx.restore();
|
|
273
|
+
}
|
|
274
|
+
atualizaCutBorder() {
|
|
275
|
+
if (this.cropBorder && this.cropBorder.nativeElement) {
|
|
276
|
+
const cropBorderElement = this.cropBorder.nativeElement;
|
|
277
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
278
|
+
cropBorderElement.style.borderRadius = this.radius() * 2 + 'px';
|
|
279
|
+
cropBorderElement.style.width = this.radius() * 2 + 'px';
|
|
280
|
+
cropBorderElement.style.height = this.radius() * 2 + 'px';
|
|
281
|
+
}
|
|
282
|
+
else {
|
|
283
|
+
cropBorderElement.style.width = this.cutWidth() + 'px';
|
|
284
|
+
cropBorderElement.style.height = this.cutHeight() + 'px';
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
//#endregion
|
|
289
|
+
//#region handlers de eventos
|
|
290
|
+
onZoomChange(e) {
|
|
291
|
+
const prevZoom = this.scaleFactorReadOnly();
|
|
292
|
+
const zoomValue = e.value;
|
|
293
|
+
const direction = prevZoom < zoomValue ? 'up' : 'down';
|
|
294
|
+
this.ajustarZoom(direction);
|
|
295
|
+
}
|
|
296
|
+
initializeMouseWheel(canvas, ctx, img) {
|
|
297
|
+
canvas.onwheel = (e) => {
|
|
298
|
+
e.preventDefault();
|
|
299
|
+
//? Recuperando os valores de zoom e delta atuais
|
|
300
|
+
const direction = e.deltaY > 0 ? 'down' : 'up';
|
|
301
|
+
this.ajustarZoom(direction);
|
|
302
|
+
};
|
|
303
|
+
}
|
|
304
|
+
ajustarZoom(direction) {
|
|
305
|
+
const prevScaleFactor = this.scaleFactorReadOnly();
|
|
306
|
+
const delta = direction == 'down' ? -0.01 : 0.01;
|
|
307
|
+
let scaleFactor = Math.min((prevScaleFactor + (delta * 1.2)), 3);
|
|
308
|
+
const newHeight = this._Image().height * scaleFactor;
|
|
309
|
+
const newWidth = this._Image().width * scaleFactor;
|
|
310
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
311
|
+
if (newHeight < this.radius() * 2 || newWidth < this.radius() * 2) {
|
|
312
|
+
scaleFactor = prevScaleFactor;
|
|
313
|
+
}
|
|
314
|
+
else {
|
|
315
|
+
this.scaleFactor.set(scaleFactor);
|
|
316
|
+
this.setScaleFactorModel(scaleFactor);
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
else {
|
|
320
|
+
if (newHeight < this.cutHeight() || newWidth < this.cutWidth()) {
|
|
321
|
+
scaleFactor = prevScaleFactor;
|
|
322
|
+
}
|
|
323
|
+
else {
|
|
324
|
+
this.scaleFactor.set(scaleFactor);
|
|
325
|
+
this.setScaleFactorModel(scaleFactor);
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
this.scaleFactor.set(scaleFactor);
|
|
329
|
+
this.setScaleFactorModel(scaleFactor);
|
|
330
|
+
const canvas = this.cropCanvas.nativeElement;
|
|
331
|
+
const canvasWidth = canvas.offsetWidth;
|
|
332
|
+
const canvasHeight = canvas.offsetHeight;
|
|
333
|
+
const centerX = canvasWidth / 2;
|
|
334
|
+
const centerY = canvasHeight / 2;
|
|
335
|
+
const newCanvasX = centerX - (centerX - this.imageX) * (scaleFactor / prevScaleFactor);
|
|
336
|
+
const newCanvasY = centerY - (centerY - this.imageY) * (scaleFactor / prevScaleFactor);
|
|
337
|
+
this.imageX = newCanvasX;
|
|
338
|
+
this.imageY = newCanvasY;
|
|
339
|
+
if (!this.verificaImagemdentroDosLimiteXmin()) {
|
|
340
|
+
const diffXmin = this.retornaDiffXmax();
|
|
341
|
+
this.moveImage(this.imageX - diffXmin);
|
|
342
|
+
}
|
|
343
|
+
if (!this.verificaImagemdentroDosLimiteXmax()) {
|
|
344
|
+
const diffXmax = this.retornaDiffXmin();
|
|
345
|
+
this.moveImage(this.imageX + diffXmax);
|
|
346
|
+
}
|
|
347
|
+
if (!this.verificaImagemdentroDosLimiteYmax()) {
|
|
348
|
+
const diffYmax = this.retornaDiffYmin();
|
|
349
|
+
this.moveImage(undefined, this.imageY + diffYmax);
|
|
350
|
+
}
|
|
351
|
+
if (!this.verificaImagemdentroDosLimiteYmin()) {
|
|
352
|
+
const diffYmin = this.retornaDiffYmax();
|
|
353
|
+
this.moveImage(undefined, this.imageY - diffYmin);
|
|
354
|
+
}
|
|
355
|
+
this.redrawCanvas(this.ctx(), this._Image());
|
|
356
|
+
}
|
|
357
|
+
verificaImagemdentroDosLimiteXmin() {
|
|
358
|
+
const newImageX = this.imageX;
|
|
359
|
+
return newImageX < this.xMin();
|
|
360
|
+
}
|
|
361
|
+
verificaImagemdentroDosLimiteXmax() {
|
|
362
|
+
const newImageX = this.imageX;
|
|
363
|
+
return newImageX > this.xMax();
|
|
364
|
+
}
|
|
365
|
+
verificaImagemdentroDosLimiteYmin() {
|
|
366
|
+
const newImageY = this.imageY;
|
|
367
|
+
return newImageY < this.yMin();
|
|
368
|
+
}
|
|
369
|
+
verificaImagemdentroDosLimiteYmax() {
|
|
370
|
+
const newImageY = this.imageY;
|
|
371
|
+
return newImageY > this.yMax();
|
|
372
|
+
}
|
|
373
|
+
retornaDiffXmin() {
|
|
374
|
+
const newImageX = this.imageX;
|
|
375
|
+
return this.xMin() - newImageX;
|
|
376
|
+
}
|
|
377
|
+
retornaDiffXmax() {
|
|
378
|
+
const newImageX = this.imageX;
|
|
379
|
+
return newImageX - this.xMax();
|
|
380
|
+
}
|
|
381
|
+
retornaDiffYmin() {
|
|
382
|
+
const newImageY = this.imageY;
|
|
383
|
+
return this.yMin() - newImageY;
|
|
384
|
+
}
|
|
385
|
+
retornaDiffYmax() {
|
|
386
|
+
const newImageY = this.imageY;
|
|
387
|
+
return newImageY - this.yMax();
|
|
388
|
+
}
|
|
389
|
+
initializeImageMove(canvas, ctx, img) {
|
|
390
|
+
let isDragging = false;
|
|
391
|
+
canvas.style.cursor = 'grab';
|
|
392
|
+
canvas.onmousedown = (e) => {
|
|
393
|
+
isDragging = true;
|
|
394
|
+
canvas.style.cursor = 'grabbing';
|
|
395
|
+
this.startX = e.offsetX - this.imageX;
|
|
396
|
+
this.startY = e.offsetY - this.imageY;
|
|
397
|
+
};
|
|
398
|
+
canvas.onmouseup = () => {
|
|
399
|
+
isDragging = false;
|
|
400
|
+
canvas.style.cursor = 'grab';
|
|
401
|
+
};
|
|
402
|
+
canvas.onmouseout = () => {
|
|
403
|
+
isDragging = false;
|
|
404
|
+
canvas.style.cursor = 'grab';
|
|
405
|
+
};
|
|
406
|
+
canvas.onmousemove = (e) => {
|
|
407
|
+
if (isDragging) {
|
|
408
|
+
// Calcula o novo posicionamento da imagem
|
|
409
|
+
let newImageX = e.offsetX - this.startX;
|
|
410
|
+
let newImageY = e.offsetY - this.startY;
|
|
411
|
+
if (newImageX < this.xMin() && newImageX > this.xMax()) {
|
|
412
|
+
this.moveImage(newImageX);
|
|
413
|
+
}
|
|
414
|
+
if (newImageY < this.yMin() && newImageY > this.yMax()) {
|
|
415
|
+
this.moveImage(undefined, newImageY);
|
|
416
|
+
}
|
|
417
|
+
this.redrawCanvas(ctx, img);
|
|
418
|
+
}
|
|
419
|
+
};
|
|
420
|
+
}
|
|
421
|
+
moveImage(dx, dy) {
|
|
422
|
+
this.imageX = (dx ?? this.imageX);
|
|
423
|
+
this.imageY = (dy ?? this.imageY);
|
|
424
|
+
}
|
|
425
|
+
//#endregion
|
|
426
|
+
//#region Métodos de corte e fechamento do modal
|
|
427
|
+
cropImage() {
|
|
428
|
+
const ctx = this.ctx();
|
|
429
|
+
if (ctx) {
|
|
430
|
+
const X = ((this.canvas().width - this._Image().width * this.scaleFactorReadOnly()) / 2);
|
|
431
|
+
const Y = ((this.canvasHeight() / 2) - (this._Image().height * this.scaleFactorReadOnly()) / 2);
|
|
432
|
+
// Recuperar o imageData da imagem que está na área de corte
|
|
433
|
+
let imageData;
|
|
434
|
+
const cropX = (ctx.canvas.width - this.cutWidth()) / 2;
|
|
435
|
+
const cropY = (ctx.canvas.height - this.cutHeight()) / 2;
|
|
436
|
+
if (this._variaviesDialog()?.indCircle) {
|
|
437
|
+
// Create a new canvas with the same dimensions as the circle
|
|
438
|
+
const circleCanvas = document.createElement('canvas');
|
|
439
|
+
const circleCtx = circleCanvas.getContext('2d');
|
|
440
|
+
circleCanvas.width = this.radius() * 2;
|
|
441
|
+
circleCanvas.height = this.radius() * 2;
|
|
442
|
+
// Draw the circle on the new canvas
|
|
443
|
+
circleCtx.beginPath();
|
|
444
|
+
circleCtx.arc(this.radius(), this.radius(), this.radius(), 0, 2 * Math.PI);
|
|
445
|
+
circleCtx.fill();
|
|
446
|
+
// Set the composite operation to 'destination-in' to cut out the circle
|
|
447
|
+
ctx.globalCompositeOperation = 'destination-in';
|
|
448
|
+
// Draw the image on the main canvas using the circle as a mask
|
|
449
|
+
ctx.drawImage(circleCanvas, cropX, cropY);
|
|
450
|
+
// Reset the composite operation to the default value
|
|
451
|
+
ctx.globalCompositeOperation = 'source-over';
|
|
452
|
+
// Get the image data of the cropped circle
|
|
453
|
+
imageData = ctx.getImageData(cropX, cropY, this.radius() * 2, this.radius() * 2);
|
|
454
|
+
}
|
|
455
|
+
else {
|
|
456
|
+
imageData = ctx.getImageData(cropX, cropY, this.cutWidth(), this.cutHeight());
|
|
457
|
+
}
|
|
458
|
+
// Obter os valores de width e height da variável _variaviesDialog
|
|
459
|
+
const variaveisDialog = this._variaviesDialog();
|
|
460
|
+
if (variaveisDialog) {
|
|
461
|
+
const { width, height } = variaveisDialog;
|
|
462
|
+
// Criar um novo canvas e contexto com os valores de width e height
|
|
463
|
+
const newCanvas = document.createElement('canvas');
|
|
464
|
+
const newCtx = newCanvas.getContext('2d');
|
|
465
|
+
newCanvas.width = width;
|
|
466
|
+
newCanvas.height = height;
|
|
467
|
+
if (newCtx) {
|
|
468
|
+
// Redimensionar a imagem para o tamanho especificado
|
|
469
|
+
const tempCanvas = document.createElement('canvas');
|
|
470
|
+
const tempCtx = tempCanvas.getContext('2d');
|
|
471
|
+
tempCanvas.width = this.cutWidth();
|
|
472
|
+
tempCanvas.height = this.cutHeight();
|
|
473
|
+
tempCtx.putImageData(imageData, 0, 0);
|
|
474
|
+
newCtx.drawImage(tempCanvas, 0, 0, width, height);
|
|
475
|
+
window.dispatchEvent(new CustomEvent('cropImage', { detail: {
|
|
476
|
+
result: Promise.resolve(newCanvas.toDataURL()),
|
|
477
|
+
status: true
|
|
478
|
+
} }));
|
|
479
|
+
this.closeCropModal();
|
|
480
|
+
}
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
}
|
|
484
|
+
closeCropModal() {
|
|
485
|
+
this.dialogInstance.close();
|
|
486
|
+
}
|
|
487
|
+
cancel() {
|
|
488
|
+
window.dispatchEvent(new CustomEvent('cropImage', { detail: {
|
|
489
|
+
result: Promise.resolve('cancel'),
|
|
490
|
+
status: false
|
|
491
|
+
} }));
|
|
492
|
+
this.closeCropModal();
|
|
493
|
+
}
|
|
494
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: SimpleComponent, deps: [{ token: i1.DynamicDialogRef }, { token: i1.DialogService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
495
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.8", type: SimpleComponent, isStandalone: true, selector: "app-simple-component", host: { listeners: { "window:resize": "onWindowResize($event)" } }, viewQueries: [{ propertyName: "cropCanvas", first: true, predicate: ["cropCanvas"], descendants: true }, { propertyName: "cropBorder", first: true, predicate: ["cropBorder"], descendants: true }], ngImport: i0, template: `
|
|
496
|
+
<div class="modal-content">
|
|
497
|
+
<div class="crop-container">
|
|
498
|
+
<canvas #cropCanvas class="crop-canvas"></canvas>
|
|
499
|
+
<div #cropBorder class="crop-border"></div>
|
|
500
|
+
</div>
|
|
501
|
+
|
|
502
|
+
<div class="w-full mt-2">
|
|
503
|
+
<kv-label label="Zoom" style="text-align: left;" />
|
|
504
|
+
<p-slider [(ngModel)]="this.scaleFactorModel" [step]="0.000001" [max]="3" (onChange)="onZoomChange($event)"></p-slider>
|
|
505
|
+
</div>
|
|
506
|
+
|
|
507
|
+
<div class="button-container">
|
|
508
|
+
<kv-button label="Cancelar" severity="tertiary" icon="close" (click)="cancel()"></kv-button>
|
|
509
|
+
<kv-button label="Recortar" severity="primary" icon="crop" (click)="cropImage()" class="ml-2"></kv-button>
|
|
510
|
+
</div>
|
|
511
|
+
|
|
512
|
+
</div>
|
|
513
|
+
|
|
514
|
+
|
|
515
|
+
|
|
516
|
+
`, isInline: true, styles: ["img{border-style:solid;border-width:2px;border-radius:50%;object-fit:cover}i{font-size:22px}.botao-alterar-foto{border-style:none;width:max-content;background-color:#002542;color:#000;font-weight:700;cursor:pointer;padding:7px 9px}.botao-alterar-foto i{color:#fff}.botao-excluir-foto{border-style:none;width:max-content;background-color:#002542;color:#000;font-weight:700;cursor:pointer;padding:7px 9px}.botao-excluir-foto i{color:#fff}.btn{position:absolute;top:87%;left:87%;transform:translate(-50%,-50%)}.image-container{display:inline-block;position:relative}.modal-content{position:relative;background-color:#fefefe;margin:1% auto;padding:20px;width:80%;max-width:fit-content;text-align:center;border-radius:12px}.crop-container{display:flex;justify-content:center;align-items:center}.crop-canvas{border:1px solid #d1d5db;border-radius:12px;width:100%;height:auto}.crop-border{position:absolute;border:2px dashed #9ca3af;pointer-events:none}.zoom-container{padding-top:20px;padding-bottom:20px}.button-container{display:flex;align-items:center;justify-content:flex-end;flex-direction:row;margin-top:24px}\n"], dependencies: [{ kind: "ngmodule", type: DialogModule }, { kind: "ngmodule", type: KvModalModule }, { kind: "ngmodule", type: SliderModule }, { kind: "component", type: i2.Slider, selector: "p-slider", inputs: ["animate", "disabled", "min", "max", "orientation", "step", "range", "style", "styleClass", "ariaLabel", "ariaLabelledBy", "tabindex", "autofocus"], outputs: ["onChange", "onSlideEnd"] }, { kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ButtonModule }, { kind: "ngmodule", type: FileUploadModule }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "ngmodule", type: KvButtonModule }, { kind: "component", type: i4.KvButtonComponent, selector: "kv-button", inputs: ["label", "icon", "loading", "disabled", "severity", "size", "fullWidth"], outputs: ["onClick"] }, { kind: "ngmodule", type: KvLabelModule }, { kind: "component", type: i5.KvLabelComponent, selector: "kv-label", inputs: ["componentId", "label"] }] }); }
|
|
517
|
+
}
|
|
518
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.8", ngImport: i0, type: SimpleComponent, decorators: [{
|
|
519
|
+
type: Component,
|
|
520
|
+
args: [{ selector: 'app-simple-component', template: `
|
|
521
|
+
<div class="modal-content">
|
|
522
|
+
<div class="crop-container">
|
|
523
|
+
<canvas #cropCanvas class="crop-canvas"></canvas>
|
|
524
|
+
<div #cropBorder class="crop-border"></div>
|
|
525
|
+
</div>
|
|
526
|
+
|
|
527
|
+
<div class="w-full mt-2">
|
|
528
|
+
<kv-label label="Zoom" style="text-align: left;" />
|
|
529
|
+
<p-slider [(ngModel)]="this.scaleFactorModel" [step]="0.000001" [max]="3" (onChange)="onZoomChange($event)"></p-slider>
|
|
530
|
+
</div>
|
|
531
|
+
|
|
532
|
+
<div class="button-container">
|
|
533
|
+
<kv-button label="Cancelar" severity="tertiary" icon="close" (click)="cancel()"></kv-button>
|
|
534
|
+
<kv-button label="Recortar" severity="primary" icon="crop" (click)="cropImage()" class="ml-2"></kv-button>
|
|
535
|
+
</div>
|
|
536
|
+
|
|
537
|
+
</div>
|
|
538
|
+
|
|
539
|
+
|
|
540
|
+
|
|
541
|
+
`, standalone: true, imports: [
|
|
542
|
+
DialogModule,
|
|
543
|
+
KvModalModule,
|
|
544
|
+
SliderModule,
|
|
545
|
+
CommonModule,
|
|
546
|
+
ButtonModule,
|
|
547
|
+
FileUploadModule,
|
|
548
|
+
SliderModule,
|
|
549
|
+
FormsModule,
|
|
550
|
+
ReactiveFormsModule,
|
|
551
|
+
KvButtonModule,
|
|
552
|
+
KvLabelModule,
|
|
553
|
+
], styles: ["img{border-style:solid;border-width:2px;border-radius:50%;object-fit:cover}i{font-size:22px}.botao-alterar-foto{border-style:none;width:max-content;background-color:#002542;color:#000;font-weight:700;cursor:pointer;padding:7px 9px}.botao-alterar-foto i{color:#fff}.botao-excluir-foto{border-style:none;width:max-content;background-color:#002542;color:#000;font-weight:700;cursor:pointer;padding:7px 9px}.botao-excluir-foto i{color:#fff}.btn{position:absolute;top:87%;left:87%;transform:translate(-50%,-50%)}.image-container{display:inline-block;position:relative}.modal-content{position:relative;background-color:#fefefe;margin:1% auto;padding:20px;width:80%;max-width:fit-content;text-align:center;border-radius:12px}.crop-container{display:flex;justify-content:center;align-items:center}.crop-canvas{border:1px solid #d1d5db;border-radius:12px;width:100%;height:auto}.crop-border{position:absolute;border:2px dashed #9ca3af;pointer-events:none}.zoom-container{padding-top:20px;padding-bottom:20px}.button-container{display:flex;align-items:center;justify-content:flex-end;flex-direction:row;margin-top:24px}\n"] }]
|
|
554
|
+
}], ctorParameters: () => [{ type: i1.DynamicDialogRef }, { type: i1.DialogService }], propDecorators: { cropCanvas: [{
|
|
555
|
+
type: ViewChild,
|
|
556
|
+
args: ['cropCanvas']
|
|
557
|
+
}], cropBorder: [{
|
|
558
|
+
type: ViewChild,
|
|
559
|
+
args: ['cropBorder']
|
|
560
|
+
}], onWindowResize: [{
|
|
561
|
+
type: HostListener,
|
|
562
|
+
args: ['window:resize', ['$event']]
|
|
563
|
+
}] } });
|
|
564
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image.cutter.service.js","sourceRoot":"","sources":["../../../../../../projects/keevo-components/src/lib/api/services/image.cutter.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ComponentRef, computed, ElementRef, HostListener, Injectable, signal, ViewChild } from '@angular/core';\r\nimport { DialogService, DynamicDialogComponent, DynamicDialogRef } from 'primeng/dynamicdialog';\r\nimport { DialogModule } from 'primeng/dialog';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class ImageCutterService {\r\n\r\n  constructor(\r\n    private readonly dialogservice: DialogService\r\n  ) { }\r\n\r\n  cut(imageBase64: string, width: number, height: number, indCircle: boolean = false): Promise<string> {\r\n    const cutConfig = { imageBase64, width, height, indCircle };\r\n    this.abrirModal(cutConfig);\r\n\r\n    return new Promise<string>((resolve, reject) => {\r\n      window.addEventListener('cropImage', (event: any) => {\r\n        if(event.detail.status == true ) {\r\n          resolve(event.detail.result);\r\n        }else{\r\n          reject('Operacao cancelada.');\r\n        }\r\n      });\r\n    });\r\n  }\r\n\r\n\r\n  abrirModal(cutConfig: { imageBase64: string, width: number, height: number, indCircle: boolean }) {\r\n    const params =       {\r\n      closable: true,\r\n      maximizable: false,\r\n      popup: true,\r\n      width: '75vw',\r\n      header: `Cortar` ,\r\n      id: { imagem: cutConfig.imageBase64, width: cutConfig.width, height: cutConfig.height, indCircle: cutConfig.indCircle },\r\n      styleClass: 'modal'\r\n    }\r\n    const dialogRef: DynamicDialogRef = FormService.openDialog(this.dialogservice, SimpleComponent, ()=>{}, params);\r\n  }\r\n}\r\n\r\n\r\n// ? Componente de modal usado no cutter\r\nimport { SliderModule } from 'primeng/slider';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\r\nimport { ButtonModule } from 'primeng/button';\r\nimport { FileUploadModule } from 'primeng/fileupload';\r\nimport { KvLabelModule } from '../../components/kv-label/kv-label.module';\r\nimport { FormService } from './form.service';\r\nimport { KvModalModule } from '../../components/kv-modal/kv-modal.module';\r\nimport { KvButtonModule } from '../../components/kv-button/kv-button.module';\r\n\r\n@Component({\r\n  selector: 'app-simple-component',\r\n  template: `\r\n      <div class=\"modal-content\">\r\n        <div class=\"crop-container\">\r\n          <canvas #cropCanvas class=\"crop-canvas\"></canvas>\r\n          <div #cropBorder class=\"crop-border\"></div>\r\n        </div>\r\n\r\n        <div class=\"w-full mt-2\">\r\n          <kv-label label=\"Zoom\" style=\"text-align: left;\" />\r\n          <p-slider [(ngModel)]=\"this.scaleFactorModel\" [step]=\"0.000001\"  [max]=\"3\"  (onChange)=\"onZoomChange($event)\"></p-slider>\r\n        </div>\r\n\r\n          <div class=\"button-container\">\r\n          <kv-button label=\"Cancelar\" severity=\"tertiary\" icon=\"close\" (click)=\"cancel()\"></kv-button>\r\n          <kv-button label=\"Recortar\" severity=\"primary\" icon=\"crop\" (click)=\"cropImage()\" class=\"ml-2\"></kv-button>\r\n        </div>\r\n\r\n      </div>\r\n\r\n\r\n\r\n  `,\r\n  styles: [`\r\n\r\nimg {\r\n  border-style: solid;\r\n  border-width: 2px;\r\n  border-radius: 50%;\r\n  object-fit: cover;\r\n}\r\n\r\ni {\r\n  font-size: 22px;\r\n}\r\n\r\n.botao-alterar-foto {\r\n  border-style: none;\r\n  width: max-content;\r\n  background-color: #002542;\r\n  color: black;\r\n  font-weight: bold;\r\n  cursor: pointer;\r\n  padding: 7px 9px;\r\n}\r\n\r\n.botao-alterar-foto i {\r\n  color: #fff;\r\n}\r\n\r\n.botao-excluir-foto {\r\n  border-style: none;\r\n  width: max-content;\r\n  background-color: #002542;\r\n  color: black;\r\n  font-weight: bold;\r\n  cursor: pointer;\r\n  padding: 7px 9px;\r\n}\r\n\r\n.botao-excluir-foto i {\r\n  color: #fff;\r\n}\r\n\r\n.btn {\r\n  position: absolute;\r\n  top: 87%;\r\n  left: 87%;\r\n  transform: translate(-50%, -50%);\r\n}\r\n\r\n.image-container {\r\n  display: inline-block;\r\n  position: relative;\r\n}\r\n\r\n.modal-content {\r\n  position: relative;\r\n  background-color: #fefefe;\r\n  margin: 1% auto;\r\n  padding: 20px;\r\n  width: 80%;\r\n  max-width: fit-content;\r\n  text-align: center;\r\n  border-radius: 12px;\r\n}\r\n\r\n.crop-container {\r\n  display: flex;\r\n  justify-content: center;\r\n  align-items: center;\r\n}\r\n\r\n.crop-canvas {\r\n  border: 1px solid #d1d5db;\r\n  border-radius: 12px;\r\n  width: 100%;\r\n  height: auto;\r\n}\r\n\r\n.crop-border {\r\n  position: absolute;\r\n  border: 2px dashed #9ca3af;\r\n  pointer-events: none;\r\n}\r\n\r\n.zoom-container {\r\n  padding-top: 20px;\r\n  padding-bottom: 20px;\r\n}\r\n\r\n.button-container {\r\n  display: flex;\r\n  align-items: center;\r\n  justify-content: flex-end;\r\n  flex-direction: row;\r\n  margin-top: 24px;\r\n}\r\n\r\n\r\n\r\n  `],\r\n  standalone: true,\r\n  imports: [\r\n    DialogModule,\r\n    KvModalModule,\r\n    SliderModule,\r\n    CommonModule,\r\n    ButtonModule,\r\n    FileUploadModule,\r\n    SliderModule,\r\n    FormsModule,\r\n    ReactiveFormsModule,\r\n    KvButtonModule,\r\n    KvLabelModule,\r\n\r\n  ]\r\n})\r\n\r\nclass SimpleComponent {\r\n\r\n  constructor(private dynamicDialogRef: DynamicDialogRef, private readonly dialogService: DialogService) {\r\n    this.atualizaTamanhoTela()\r\n  }\r\n\r\n\r\n  // ? Variavies que recebem os valores do dialog\r\n  _variaviesDialog = signal<{ imagem: string, width: number, height: number, indCircle: boolean }|null>(null)\r\n\r\n\r\n\r\n    //? Referência para o elemento que contém a imagem e canvas\r\n    @ViewChild('cropCanvas') cropCanvas!: ElementRef<HTMLCanvasElement>;\r\n    @ViewChild('cropBorder') cropBorder!: ElementRef;\r\n\r\n    //? Referências para o contexto e canvas\r\n    private canvas = signal<HTMLCanvasElement|null>(null);\r\n    private ctx =  signal<CanvasRenderingContext2D|null>(null);\r\n    private _Image = signal<HTMLImageElement>(new Image());\r\n\r\n    //? Coordenadas iniciais e finais do desenho\r\n    private startX = 0;\r\n    private startY = 0;\r\n\r\n    //? Referência para o modal\r\n    private dialogInstance!:DynamicDialogComponent;\r\n\r\n    //? Fator de escala da imagem e zoom\r\n    scaleFactorReadOnly = computed<number>(()=> {\r\n      return Math.min(this.scaleFactor()!, 3);\r\n    })\r\n    private scaleFactor = signal<number|null>(null);\r\n    setScaleFactorModel(valor:number){\r\n      this.scaleFactorModel = valor\r\n      this.scaleFactor.set(this.scaleFactorModel)\r\n    }\r\n    scaleFactorModel = 0;\r\n\r\n    private imageX: number = 0;\r\n    private imageY: number = 0;\r\n\r\n\r\n    //#region Propriedades computadas de tamanho do canvas e da imagem\r\n      //? Largura e altura do canvas baseado na tamanho da tela\r\n      canvasWidth = computed<number | null>(() => this.widthTela()*0.5);\r\n      canvasHeight = computed<number | null>(() => this.heigthTela()*0.5);\r\n\r\n      //? Largura e altura da área de corte\r\n      cutWidth = computed<number | null>(() => {\r\n        if(this._variaviesDialog()?.indCircle){\r\n          return this.radius()! * 2;\r\n        }else{\r\n          const canvasWidth = this.canvasWidth()!;\r\n          const maxWidth = canvasWidth * 0.8;\r\n          const proportionalHeight = maxWidth * this.proporcaoImagem()!;\r\n          return proportionalHeight > this.canvasHeight()! * 0.8 ? this.canvasHeight()! * 0.8 / this.proporcaoImagem()! : maxWidth;\r\n        }\r\n\r\n      });\r\n      cutHeight = computed<number | null>(() => {\r\n        if(this._variaviesDialog()?.indCircle){\r\n          return this.radius()! * 2;\r\n        }else{\r\n          const canvasHeight = this.canvasHeight()!;\r\n          const maxHeight = canvasHeight * 0.8;\r\n          const proportionalWidth = maxHeight / this.proporcaoImagem()!;\r\n          return proportionalWidth > this.canvasWidth()! * 0.8 ? this.canvasWidth()! * 0.8 * this.proporcaoImagem()! : maxHeight;\r\n        }\r\n\r\n      });\r\n\r\n      //? Largura e altura da imagem no canvas (baseado na escala) - tamanho real da imagem\r\n      canvaImageWidth = computed<number>(()=> {\r\n        return this._Image().width * this.scaleFactorReadOnly()!\r\n      })\r\n      canvaImageHeight = computed<number>(()=> {\r\n        return this._Image().height * this.scaleFactorReadOnly()!\r\n      })\r\n      proporcaoImagem = computed<number | null>(() => {\r\n        return this._variaviesDialog()?.height! / this._variaviesDialog()?.width!;\r\n      });\r\n\r\n      radius = computed<number | null>(() => {\r\n        const minSize =   Math.min(this.canvasHeight()!*0.8, this.canvasWidth()!*0.8 );\r\n        return minSize/2;\r\n      });\r\n\r\n    //#endregion\r\n\r\n    // #region Limites de movimentação da imagem\r\n    //? Limites de movimentação da imagem\r\n      indCircle = computed<boolean>(()=> {\r\n        return this._variaviesDialog()?.indCircle!\r\n      })\r\n\r\n      xMin = computed<number>(()=> {\r\n        return !this._variaviesDialog()?.indCircle! ? (this.canvasWidth()! -  this.cutWidth()!)/2 : (this.canvasWidth()! -  (this.radius()!*2))/2\r\n      })\r\n      yMin = computed<number>(()=> {\r\n        return !this._variaviesDialog()?.indCircle! ? (this.canvasHeight()! -  this.cutHeight()!)/2 : (this.canvasHeight()! -  (this.radius()!*2))/2\r\n      })\r\n\r\n      xMax = computed<number>(()=> {\r\n        return !this._variaviesDialog()?.indCircle! ? -( this.canvaImageWidth()! - (this.xMin()!) - this.cutWidth()!) : -( this.canvaImageWidth()! - (this.xMin()!) - (this.radius()!*2))\r\n      })\r\n      yMax = computed<number>(()=> {\r\n        return !this._variaviesDialog()?.indCircle! ? -( this.canvaImageHeight()! - (this.yMin()!) - this.cutHeight()!) : -( this.canvaImageHeight()! - (this.yMin()!) - (this.radius()!*2))\r\n      })\r\n    // #endregion\r\n\r\n\r\n\r\n    //? Proporção da imagem\r\n\r\n    widthTela = signal<number>(0)\r\n    heigthTela = signal<number>(0)\r\n\r\n\r\n\r\n  @HostListener('window:resize', ['$event'])\r\n  onWindowResize() {\r\n    this.atualizaTamanhoTela()\r\n    this.updateCanvasSize();\r\n    this.atualizaCutBorder();\r\n  }\r\n\r\n  atualizaTamanhoTela(){\r\n    this.widthTela.set(window.innerWidth);\r\n    this.heigthTela.set(window.innerHeight);\r\n  }\r\n\r\n  ngAfterViewInit() {\r\n    this.recuperarVariaviesDialog()\r\n    this.inciarReferencias()\r\n    this.iniciarCanvas()\r\n  }\r\n  inciarReferencias(){\r\n    this.canvas.set(this.cropCanvas.nativeElement);\r\n    this.ctx.set(this.canvas()!.getContext('2d'))\r\n    this._Image().src = this._variaviesDialog()?.imagem!\r\n  }\r\n  iniciarCanvas(){\r\n      this._Image().onload = () => {\r\n\r\n      //? Define o tamanho do canvas\r\n      this.canvas()!.height = this.canvasHeight()!;\r\n      this.canvas()!.width = this.canvasWidth()!;\r\n\r\n\r\n      this.ajustarZommEPosicaoInicial()\r\n\r\n\r\n      this.initializeImageMove(this.canvas()!, this.ctx(), this._Image());\r\n      this.initializeMouseWheel(this.canvas()!, this.ctx(), this._Image());\r\n\r\n      this.redrawCanvas(this.ctx(), this._Image());\r\n\r\n    };\r\n  }\r\n\r\n  ajustarZommEPosicaoInicial(){\r\n    const cutWidth = this.cutWidth()!;\r\n    const cutHeight = this.cutHeight()!;\r\n    const imageWidth = this._Image().width;\r\n    const imageHeight = this._Image().height;\r\n\r\n    //? Ajusta possição e zoom inicial\r\n\r\n    if (this._variaviesDialog()?.indCircle) {\r\n      const scaleFactor = (this.radius()! * 2) / Math.min(imageWidth, imageHeight);\r\n      this.setScaleFactorModel(scaleFactor);\r\n      this.scaleFactor.set(scaleFactor);\r\n    } else {\r\n      const scaleFactor = this.cutHeight()! === this.cutWidth()!\r\n        ? this._Image().width > this._Image().height\r\n          ? this.cutHeight()! / imageHeight\r\n          : this.cutWidth()! / imageWidth\r\n        : this.cutHeight()! > this.cutWidth()!\r\n          ? this.cutHeight()! / imageHeight\r\n          : this.cutWidth()! / imageWidth;\r\n      this.setScaleFactorModel(scaleFactor);\r\n      this.scaleFactor.set(scaleFactor);\r\n    }\r\n\r\n    this.imageX = Math.max((this.canvas()!.width - imageWidth * this.scaleFactorReadOnly()!) / 2, this.xMin()!);\r\n    this.imageY = Math.max((this.canvasHeight()! - imageHeight * this.scaleFactorReadOnly()!) / 2, this.yMin()!);\r\n\r\n  }\r\n\r\n  //#region Métodos de recuperação de dados\r\n  recuperaInstanciaDialog(): DynamicDialogComponent {\r\n    const arr: Map<DynamicDialogRef<any>, ComponentRef<any>> = this.dialogService.dialogComponentRefMap\r\n    const entries = arr.entries();\r\n    const firstEntry = entries.next().value;\r\n    const [firstKey, firstValue] = firstEntry;\r\n    const instance = this.dialogService.getInstance(firstKey)\r\n    this.dialogInstance = instance\r\n    return instance\r\n  }\r\n\r\n  recuperarVariaviesDialog(): void{\r\n    this._variaviesDialog.set(this.recuperaInstanciaDialog().data.id)\r\n  }\r\n  //#endregion\r\n\r\n\r\n  //#region Métodos de atualizacao do canvas\r\n  //? Atualiza o tamanho do canvas\r\n  updateCanvasSize() {\r\n\r\n    this.imageX = (this.canvas()!.width - this._Image().width * this.scaleFactorReadOnly()!) / 2;\r\n    this.imageY = (this.canvasHeight()! / 2) - (this._Image().height * this.scaleFactorReadOnly()!) / 2;\r\n\r\n\r\n\r\n    this.canvas()!.height = this.canvasHeight()!;\r\n    this.canvas()!.width = this.canvasWidth()!;\r\n\r\n    this.redrawCanvas(this.ctx(), this._Image());\r\n  }\r\n\r\n  redrawCanvas(ctx: CanvasRenderingContext2D | null, img: HTMLImageElement) {\r\n    if (!ctx) return;\r\n\r\n    this.drawCropArea(ctx);\r\n\r\n    ctx.clearRect(0, 0, ctx.canvas.width, ctx.canvas.height);\r\n    ctx.filter = 'blur(5px) opacity(0.9)';\r\n    ctx.drawImage(img, this.imageX, this.imageY, img.width * this.scaleFactorReadOnly()!, img.height * this.scaleFactorReadOnly()!);\r\n    ctx.filter = 'none';\r\n\r\n    this.drawImageInCropArea(ctx, img);\r\n  }\r\n\r\n  //? Desenha a área de corte\r\n  drawCropArea(ctx: CanvasRenderingContext2D | null) {\r\n    if (!ctx) return;\r\n\r\n    this.atualizaCutBorder()\r\n\r\n    const cropX = ctx.canvas.width / 2 - this.cutWidth()! / 2;\r\n    const cropY = ctx.canvas.height / 2 - this.cutHeight()! / 2;\r\n\r\n\r\n    const cropXCircle = ctx.canvas.width / 2 - this.radius()! / 2;\r\n    const cropYCircle = ctx.canvas.height / 2 - this.radius()! / 2;\r\n\r\n    ctx.beginPath();\r\n    ctx.strokeStyle = 'transparent';\r\n    ctx.lineWidth = 2;\r\n\r\n    if(this._variaviesDialog()?.indCircle) {\r\n      ctx.arc(cropXCircle, cropYCircle, this.radius()!, 0, Math.PI * 2, true);\r\n    } else {\r\n      ctx.rect(cropX, cropY , this.cutWidth()!, this.cutHeight()!);\r\n    }\r\n\r\n    ctx.stroke();\r\n\r\n\r\n  }\r\n  //? Desenha a imagem na área de corte\r\n  drawImageInCropArea(ctx: CanvasRenderingContext2D | null, img: HTMLImageElement) {\r\n      if (!ctx) return;\r\n\r\n      const cropX = (ctx.canvas.width - this.cutWidth()!) / 2;\r\n      const cropY = (ctx.canvas.height - this.cutHeight()!) / 2;\r\n\r\n      ctx.save();\r\n      ctx.beginPath();\r\n\r\n      if(this._variaviesDialog()?.indCircle) {\r\n        ctx.arc(ctx.canvas.width/2, ctx.canvas.height/2, this.radius()!, 0, Math.PI * 2, true);\r\n\r\n      } else {\r\n        ctx.rect(cropX, cropY , this.cutWidth()!, this.cutHeight()!);\r\n      }\r\n\r\n      ctx.clip();\r\n      ctx.drawImage(img, this.imageX, this.imageY, img.width * this.scaleFactorReadOnly()!, img.height * this.scaleFactorReadOnly()!);\r\n      ctx.restore();\r\n  }\r\n  atualizaCutBorder() {\r\n    if (this.cropBorder && this.cropBorder.nativeElement) {\r\n      const cropBorderElement = this.cropBorder.nativeElement as HTMLElement;\r\n      if(this._variaviesDialog()?.indCircle){\r\n\r\n        cropBorderElement.style.borderRadius = this.radius()!*2+'px';\r\n        cropBorderElement.style.width = this.radius()!*2+'px';\r\n        cropBorderElement.style.height = this.radius()!*2+'px';\r\n      }else{\r\n        cropBorderElement.style.width = this.cutWidth()!+'px';\r\n        cropBorderElement.style.height = this.cutHeight()!+'px';\r\n      }\r\n    }\r\n  }\r\n  //#endregion\r\n\r\n\r\n  //#region handlers de eventos\r\n\r\n  onZoomChange(e: any) {\r\n    const prevZoom = this.scaleFactorReadOnly()!;\r\n    const zoomValue = e.value;\r\n    const direction: 'up' | 'down' = prevZoom < zoomValue ? 'up' : 'down';\r\n    this.ajustarZoom(direction)\r\n  }\r\n\r\n\r\n\r\n  initializeMouseWheel(canvas: HTMLCanvasElement, ctx: CanvasRenderingContext2D | null, img: HTMLImageElement) {\r\n    canvas.onwheel = (e) => {\r\n      e.preventDefault();\r\n      //? Recuperando os valores de zoom e delta atuais\r\n      const direction: 'up' | 'down' = e.deltaY > 0 ? 'down' : 'up';\r\n\r\n      this.ajustarZoom(direction)\r\n    };\r\n  }\r\n\r\n\r\n  ajustarZoom(direction: 'up' | 'down') {\r\n    const prevScaleFactor = this.scaleFactorReadOnly()!;\r\n    const delta = direction == 'down' ? -0.01 : 0.01;\r\n\r\n    let scaleFactor = Math.min((prevScaleFactor + (delta*1.2)), 3);\r\n\r\n    const newHeight = this._Image().height * scaleFactor;\r\n    const newWidth = this._Image().width * scaleFactor;\r\n\r\n    if (this._variaviesDialog()?.indCircle) {\r\n      if (newHeight < this.radius()! * 2 || newWidth < this.radius()! * 2) {\r\n        scaleFactor = prevScaleFactor;\r\n      } else {\r\n        this.scaleFactor.set(scaleFactor);\r\n        this.setScaleFactorModel(scaleFactor);\r\n      }\r\n    } else {\r\n      if (newHeight < this.cutHeight()! || newWidth < this.cutWidth()!) {\r\n        scaleFactor = prevScaleFactor;\r\n      } else {\r\n        this.scaleFactor.set(scaleFactor);\r\n        this.setScaleFactorModel(scaleFactor);\r\n      }\r\n    }\r\n\r\n    this.scaleFactor.set(scaleFactor);\r\n    this.setScaleFactorModel(scaleFactor);\r\n\r\n    const canvas = this.cropCanvas.nativeElement;\r\n    const canvasWidth = canvas.offsetWidth;\r\n    const canvasHeight = canvas.offsetHeight;\r\n    const centerX = canvasWidth / 2;\r\n    const centerY = canvasHeight / 2;\r\n    const newCanvasX = centerX - (centerX - this.imageX) * (scaleFactor / prevScaleFactor);\r\n    const newCanvasY = centerY - (centerY - this.imageY) * (scaleFactor / prevScaleFactor);\r\n    this.imageX = newCanvasX;\r\n    this.imageY = newCanvasY;\r\n\r\n    if (!this.verificaImagemdentroDosLimiteXmin()) {\r\n      const diffXmin = this.retornaDiffXmax();\r\n      this.moveImage(this.imageX - diffXmin);\r\n    }\r\n\r\n    if (!this.verificaImagemdentroDosLimiteXmax()) {\r\n      const diffXmax = this.retornaDiffXmin();\r\n      this.moveImage(this.imageX + diffXmax);\r\n    }\r\n\r\n    if (!this.verificaImagemdentroDosLimiteYmax()) {\r\n      const diffYmax = this.retornaDiffYmin();\r\n      this.moveImage(undefined, this.imageY + diffYmax);\r\n    }\r\n\r\n    if (!this.verificaImagemdentroDosLimiteYmin()) {\r\n      const diffYmin = this.retornaDiffYmax();\r\n      this.moveImage(undefined, this.imageY - diffYmin);\r\n    }\r\n\r\n    this.redrawCanvas(this.ctx()!, this._Image()!);\r\n  }\r\n\r\n\r\n\r\n\r\n  verificaImagemdentroDosLimiteXmin(): boolean {\r\n    const newImageX = this.imageX;\r\n    return newImageX < this.xMin()!\r\n  }\r\n  verificaImagemdentroDosLimiteXmax(): boolean {\r\n    const newImageX = this.imageX;\r\n    return newImageX > this.xMax()!\r\n  }\r\n  verificaImagemdentroDosLimiteYmin(): boolean {\r\n    const newImageY = this.imageY;\r\n    return newImageY < this.yMin()!\r\n  }\r\n  verificaImagemdentroDosLimiteYmax(): boolean {\r\n    const newImageY = this.imageY;\r\n    return newImageY > this.yMax()!\r\n  }\r\n\r\n  retornaDiffXmin(): number {\r\n    const newImageX = this.imageX;\r\n    return this.xMin()! - newImageX;\r\n  }\r\n  retornaDiffXmax(): number {\r\n    const newImageX = this.imageX;\r\n    return newImageX - this.xMax()!;\r\n  }\r\n  retornaDiffYmin(): number {\r\n    const newImageY = this.imageY;\r\n    return this.yMin()! - newImageY;\r\n  }\r\n  retornaDiffYmax(): number {\r\n    const newImageY = this.imageY;\r\n    return newImageY - this.yMax()!;\r\n  }\r\n\r\n\r\n  initializeImageMove(canvas: HTMLCanvasElement, ctx: CanvasRenderingContext2D | null, img: HTMLImageElement) {\r\n    let isDragging = false;\r\n\r\n    canvas.style.cursor = 'grab';\r\n\r\n    canvas.onmousedown = (e) => {\r\n      isDragging = true;\r\n      canvas.style.cursor = 'grabbing';\r\n      this.startX = e.offsetX - this.imageX;\r\n      this.startY = e.offsetY - this.imageY;\r\n    };\r\n\r\n    canvas.onmouseup = () => {\r\n      isDragging = false;\r\n      canvas.style.cursor = 'grab';\r\n    };\r\n\r\n    canvas.onmouseout = () => {\r\n      isDragging = false;\r\n      canvas.style.cursor = 'grab';\r\n    };\r\n\r\n    canvas.onmousemove = (e) => {\r\n      if (isDragging) {\r\n\r\n          // Calcula o novo posicionamento da imagem\r\n          let newImageX = e.offsetX - this.startX;\r\n          let newImageY = e.offsetY - this.startY;\r\n\r\n          if (newImageX <  this.xMin()! && newImageX > this.xMax()!) {\r\n            this.moveImage(newImageX)\r\n          }\r\n          if(newImageY < this.yMin()! && newImageY > this.yMax()!){\r\n            this.moveImage(undefined, newImageY)\r\n          }\r\n\r\n\r\n          this.redrawCanvas(ctx, img);\r\n      }\r\n    };\r\n\r\n\r\n  }\r\n\r\n  moveImage(dx?: number, dy?: number) {\r\n    this.imageX = (dx??this.imageX);\r\n    this.imageY = (dy??this.imageY);\r\n  }\r\n\r\n\r\n\r\n  //#endregion\r\n\r\n\r\n  //#region Métodos de corte e fechamento do modal\r\n  cropImage() {\r\n    const ctx = this.ctx();\r\n\r\n    if (ctx) {\r\n      const X = ((this.canvas()!.width - this._Image().width * this.scaleFactorReadOnly()!) / 2);\r\n      const Y = ((this.canvasHeight()! / 2) - (this._Image().height * this.scaleFactorReadOnly()!) / 2);\r\n\r\n      // Recuperar o imageData da imagem que está na área de corte\r\n      let imageData\r\n      const cropX = (ctx.canvas.width - this.cutWidth()!) / 2;\r\n      const cropY = (ctx.canvas.height - this.cutHeight()!) / 2;\r\n      if(this._variaviesDialog()?.indCircle){\r\n        // Create a new canvas with the same dimensions as the circle\r\n        const circleCanvas = document.createElement('canvas');\r\n        const circleCtx = circleCanvas.getContext('2d');\r\n        circleCanvas.width = this.radius()! * 2;\r\n        circleCanvas.height = this.radius()! * 2;\r\n\r\n        // Draw the circle on the new canvas\r\n        circleCtx!.beginPath();\r\n        circleCtx!.arc(this.radius()!, this.radius()!, this.radius()!, 0, 2 * Math.PI);\r\n        circleCtx!.fill();\r\n\r\n        // Set the composite operation to 'destination-in' to cut out the circle\r\n        ctx.globalCompositeOperation = 'destination-in';\r\n\r\n        // Draw the image on the main canvas using the circle as a mask\r\n        ctx.drawImage(circleCanvas, cropX, cropY);\r\n\r\n        // Reset the composite operation to the default value\r\n        ctx.globalCompositeOperation = 'source-over';\r\n\r\n        // Get the image data of the cropped circle\r\n        imageData = ctx.getImageData(cropX, cropY, this.radius()! * 2, this.radius()! * 2);\r\n      } else {\r\n        imageData = ctx.getImageData(cropX, cropY, this.cutWidth()!, this.cutHeight()!);\r\n      }\r\n\r\n\r\n      // Obter os valores de width e height da variável _variaviesDialog\r\n      const variaveisDialog = this._variaviesDialog();\r\n      if (variaveisDialog) {\r\n        const { width, height } = variaveisDialog;\r\n\r\n        // Criar um novo canvas e contexto com os valores de width e height\r\n        const newCanvas = document.createElement('canvas');\r\n        const newCtx = newCanvas.getContext('2d');\r\n        newCanvas.width = width;\r\n        newCanvas.height = height;\r\n\r\n        if (newCtx) {\r\n          // Redimensionar a imagem para o tamanho especificado\r\n          const tempCanvas = document.createElement('canvas');\r\n          const tempCtx = tempCanvas.getContext('2d');\r\n          tempCanvas.width = this.cutWidth()!;\r\n          tempCanvas.height = this.cutHeight()!;\r\n          tempCtx!.putImageData(imageData, 0, 0);\r\n\r\n          newCtx.drawImage(tempCanvas, 0, 0, width, height);\r\n\r\n          window.dispatchEvent(new CustomEvent('cropImage', { detail:{\r\n            result: Promise.resolve(newCanvas.toDataURL()),\r\n            status: true\r\n          } }));\r\n          this.closeCropModal();\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  closeCropModal() {\r\n    this.dialogInstance.close();\r\n  }\r\n\r\n  cancel(){\r\n    window.dispatchEvent(new CustomEvent('cropImage', { detail:{\r\n      result: Promise.resolve('cancel'),\r\n      status: false\r\n    } }));\r\n\r\n    this.closeCropModal()\r\n  }\r\n  //#endregion\r\n\r\n}\r\n"]}
|