valtech-components 2.0.680 → 2.0.681

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (33) hide show
  1. package/esm2022/lib/components/templates/docs-page/docs-page.component.mjs +4 -35
  2. package/esm2022/lib/components/templates/docs-page/types.mjs +1 -1
  3. package/esm2022/lib/services/auth/auth.service.mjs +2 -11
  4. package/esm2022/lib/services/auth/types.mjs +1 -1
  5. package/esm2022/lib/version.mjs +2 -2
  6. package/esm2022/public-api.mjs +1 -7
  7. package/fesm2022/valtech-components.mjs +7 -815
  8. package/fesm2022/valtech-components.mjs.map +1 -1
  9. package/lib/components/atoms/rights-footer/rights-footer.component.d.ts +1 -1
  10. package/lib/components/organisms/article/article.component.d.ts +1 -1
  11. package/lib/components/organisms/toolbar/toolbar.component.d.ts +1 -1
  12. package/lib/components/templates/docs-page/docs-page.component.d.ts +0 -3
  13. package/lib/components/templates/docs-page/types.d.ts +0 -39
  14. package/lib/services/auth/auth.service.d.ts +1 -6
  15. package/lib/services/auth/types.d.ts +0 -18
  16. package/lib/version.d.ts +1 -1
  17. package/package.json +2 -6
  18. package/public-api.d.ts +0 -4
  19. package/esm2022/lib/components/molecules/image-crop/image-crop.component.mjs +0 -174
  20. package/esm2022/lib/components/molecules/image-crop/index.mjs +0 -2
  21. package/esm2022/lib/components/organisms/avatar-upload/avatar-upload.component.mjs +0 -345
  22. package/esm2022/lib/components/organisms/avatar-upload/types.mjs +0 -15
  23. package/esm2022/lib/services/image/image.service.mjs +0 -244
  24. package/esm2022/lib/services/image/index.mjs +0 -3
  25. package/esm2022/lib/services/image/types.mjs +0 -13
  26. package/lib/components/molecules/image-crop/image-crop.component.d.ts +0 -59
  27. package/lib/components/molecules/image-crop/index.d.ts +0 -1
  28. package/lib/components/organisms/avatar-upload/avatar-upload.component.d.ts +0 -82
  29. package/lib/components/organisms/avatar-upload/types.d.ts +0 -62
  30. package/lib/services/image/image.service.d.ts +0 -76
  31. package/lib/services/image/index.d.ts +0 -2
  32. package/lib/services/image/types.d.ts +0 -74
  33. package/src/lib/services/firebase/firebase-messaging-sw.js +0 -134
@@ -1,345 +0,0 @@
1
- import { CommonModule } from '@angular/common';
2
- import { Component, computed, EventEmitter, inject, input, Output, signal, ViewChild, } from '@angular/core';
3
- import { IonIcon, IonModal, IonSpinner } from '@ionic/angular/standalone';
4
- import { addIcons } from 'ionicons';
5
- import { cameraOutline } from 'ionicons/icons';
6
- import { firstValueFrom } from 'rxjs';
7
- import { AuthService } from '../../../services/auth';
8
- import { StorageService } from '../../../services/firebase';
9
- import { I18nService } from '../../../services/i18n';
10
- import { ImageService } from '../../../services/image';
11
- import { ImageCropComponent } from '../../molecules/image-crop';
12
- import { AVATAR_UPLOAD_DEFAULTS, } from './types';
13
- import * as i0 from "@angular/core";
14
- addIcons({ cameraOutline });
15
- /**
16
- * AvatarUploadComponent
17
- *
18
- * A complete avatar upload solution with:
19
- * - Image selection from device
20
- * - Crop modal with round preview
21
- * - Automatic compression and thumbnail generation
22
- * - Upload to Firebase Storage
23
- * - Backend sync via AuthService
24
- *
25
- * @example Basic usage
26
- * ```html
27
- * <val-avatar-upload
28
- * [props]="{
29
- * currentUrl: user()?.avatarUrl,
30
- * initials: 'JD',
31
- * size: 120
32
- * }"
33
- * (uploaded)="onAvatarUploaded($event)"
34
- * (error)="onError($event)"
35
- * />
36
- * ```
37
- */
38
- export class AvatarUploadComponent {
39
- constructor() {
40
- this.imageService = inject(ImageService);
41
- this.storageService = inject(StorageService);
42
- this.authService = inject(AuthService);
43
- this.i18n = inject(I18nService);
44
- /** Component configuration */
45
- this.props = input({});
46
- /** Emitted after successful upload and backend sync */
47
- this.uploaded = new EventEmitter();
48
- /** Emitted on any error during the process */
49
- this.error = new EventEmitter();
50
- /** Emitted when upload starts */
51
- this.uploadStart = new EventEmitter();
52
- // Internal state
53
- this.loading = signal(false);
54
- this.showCropModal = signal(false);
55
- this.selectedFile = signal(null);
56
- this.previewUrl = signal(null);
57
- this.imageLoadError = signal(false);
58
- /** Merged config with defaults */
59
- this.config = computed(() => ({
60
- ...AVATAR_UPLOAD_DEFAULTS,
61
- ...this.props(),
62
- }));
63
- /** URL to display (preview takes priority over current) */
64
- this.displayUrl = computed(() => {
65
- if (this.imageLoadError())
66
- return null;
67
- return this.previewUrl() || this.config().currentUrl || null;
68
- });
69
- /** Aria label for edit button */
70
- this.editButtonLabel = computed(() => {
71
- this.i18n.lang();
72
- return this.i18n.t('changePhoto', this.config().i18nNamespace) || 'Cambiar foto';
73
- });
74
- }
75
- /** Open file picker dialog */
76
- openFilePicker() {
77
- this.fileInput.nativeElement.click();
78
- }
79
- /** Handle file selection */
80
- onFileSelected(event) {
81
- const input = event.target;
82
- const file = input.files?.[0];
83
- if (!file)
84
- return;
85
- // Reset input for same file selection
86
- input.value = '';
87
- // Validate file
88
- const validation = this.imageService.validate(file, {
89
- maxSize: this.config().maxFileSize,
90
- allowedTypes: ['image/jpeg', 'image/png', 'image/webp'],
91
- });
92
- if (!validation.valid) {
93
- this.emitError(validation.error, validation.message);
94
- return;
95
- }
96
- // Open crop modal
97
- this.selectedFile.set(file);
98
- this.showCropModal.set(true);
99
- }
100
- /** Handle crop completion */
101
- async onCropComplete(croppedBlob) {
102
- this.showCropModal.set(false);
103
- this.selectedFile.set(null);
104
- await this.processAndUpload(croppedBlob);
105
- }
106
- /** Handle crop cancel */
107
- onCropCancel() {
108
- this.showCropModal.set(false);
109
- this.selectedFile.set(null);
110
- }
111
- /** Handle crop load failure */
112
- onCropLoadFailed() {
113
- this.showCropModal.set(false);
114
- this.selectedFile.set(null);
115
- this.emitError('invalidType', this.i18n.t('loadFailed', this.config().i18nNamespace) || 'No se pudo cargar la imagen');
116
- }
117
- /** Handle image load error */
118
- onImageError() {
119
- this.imageLoadError.set(true);
120
- }
121
- /** Process cropped image and upload */
122
- async processAndUpload(croppedBlob) {
123
- this.loading.set(true);
124
- this.uploadStart.emit();
125
- try {
126
- const config = this.config();
127
- // 1. Compress image
128
- const compressed = await this.imageService.compress(croppedBlob, {
129
- maxWidth: config.maxWidth,
130
- maxHeight: config.maxWidth,
131
- quality: config.compressQuality,
132
- });
133
- // 2. Generate thumbnail
134
- const thumbnail = await this.imageService.thumbnail(compressed.blob, config.thumbnailSize);
135
- // 3. Set preview immediately
136
- this.previewUrl.set(compressed.dataUrl);
137
- this.imageLoadError.set(false);
138
- // 4. Get user ID for storage path
139
- const userId = this.authService.user()?.userId;
140
- if (!userId) {
141
- throw new Error('User not authenticated');
142
- }
143
- // 5. Upload to Firebase Storage
144
- const timestamp = Date.now();
145
- const avatarPath = `${config.storagePath}/${userId}/avatar_${timestamp}.jpg`;
146
- const thumbPath = `${config.storagePath}/${userId}/thumb_${timestamp}.jpg`;
147
- const [avatarResult, thumbResult] = await Promise.all([
148
- this.storageService.uploadAndGetUrl(avatarPath, compressed.blob),
149
- this.storageService.uploadAndGetUrl(thumbPath, thumbnail.blob),
150
- ]);
151
- // 6. Update backend
152
- await firstValueFrom(this.authService.updateAvatar({
153
- avatarUrl: avatarResult.downloadUrl,
154
- avatarThumbnail: thumbResult.downloadUrl,
155
- }));
156
- // 7. Emit success
157
- const result = {
158
- avatarUrl: avatarResult.downloadUrl,
159
- thumbnailUrl: thumbResult.downloadUrl,
160
- };
161
- this.uploaded.emit(result);
162
- }
163
- catch (err) {
164
- // Revert preview on error
165
- this.previewUrl.set(null);
166
- const message = err instanceof Error
167
- ? err.message
168
- : this.i18n.t('uploadError', this.config().i18nNamespace) || 'Error al subir la imagen';
169
- this.emitError('uploadFailed', message, err);
170
- }
171
- finally {
172
- this.loading.set(false);
173
- }
174
- }
175
- /** Emit error event */
176
- emitError(type, message, originalError) {
177
- this.error.emit({ type, message, originalError });
178
- }
179
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AvatarUploadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
180
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: AvatarUploadComponent, isStandalone: true, selector: "val-avatar-upload", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { uploaded: "uploaded", error: "error", uploadStart: "uploadStart" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }], ngImport: i0, template: `
181
- <div
182
- class="avatar-upload"
183
- [style.--avatar-size.px]="config().size"
184
- [class.avatar-upload--loading]="loading()"
185
- >
186
- <div class="avatar-container">
187
- <!-- Avatar Image or Initials -->
188
- @if (displayUrl()) {
189
- <img
190
- class="avatar-image"
191
- [src]="displayUrl()"
192
- alt="Avatar"
193
- (error)="onImageError()"
194
- />
195
- } @else {
196
- <div
197
- class="avatar-initials"
198
- [style.background-color]="config().backgroundColor"
199
- >
200
- {{ config().initials || '?' }}
201
- </div>
202
- }
203
-
204
- <!-- Edit Button -->
205
- @if (config().editable && !loading()) {
206
- <button
207
- class="edit-button"
208
- type="button"
209
- (click)="openFilePicker()"
210
- [attr.aria-label]="editButtonLabel()"
211
- >
212
- <ion-icon name="camera-outline"></ion-icon>
213
- </button>
214
- }
215
-
216
- <!-- Loading Overlay -->
217
- @if (loading()) {
218
- <div class="loading-overlay">
219
- <ion-spinner name="crescent"></ion-spinner>
220
- </div>
221
- }
222
- </div>
223
-
224
- <!-- Hidden File Input -->
225
- <input
226
- #fileInput
227
- type="file"
228
- accept="image/jpeg,image/png,image/webp"
229
- (change)="onFileSelected($event)"
230
- hidden
231
- />
232
-
233
- <!-- Crop Modal -->
234
- <ion-modal
235
- [isOpen]="showCropModal()"
236
- (didDismiss)="onCropCancel()"
237
- [breakpoints]="[0, 1]"
238
- [initialBreakpoint]="1"
239
- >
240
- <ng-template>
241
- @if (selectedFile()) {
242
- <val-image-crop
243
- [image]="selectedFile()!"
244
- [aspectRatio]="1"
245
- [roundCropper]="true"
246
- [i18nNamespace]="config().i18nNamespace"
247
- (cropComplete)="onCropComplete($event)"
248
- (cancel)="onCropCancel()"
249
- (loadFailed)="onCropLoadFailed()"
250
- />
251
- }
252
- </ng-template>
253
- </ion-modal>
254
- </div>
255
- `, isInline: true, styles: [".avatar-upload{--avatar-size: 100px;--edit-button-size: 32px;--edit-button-offset: 4px;display:inline-block}.avatar-container{position:relative;width:var(--avatar-size);height:var(--avatar-size);border-radius:50%;overflow:visible}.avatar-image{width:100%;height:100%;border-radius:50%;object-fit:cover;background-color:var(--ion-color-light)}.avatar-initials{width:100%;height:100%;border-radius:50%;display:flex;align-items:center;justify-content:center;font-size:calc(var(--avatar-size) * .4);font-weight:600;color:#fff;text-transform:uppercase;-webkit-user-select:none;user-select:none}.edit-button{position:absolute;bottom:var(--edit-button-offset);right:var(--edit-button-offset);width:var(--edit-button-size);height:var(--edit-button-size);border-radius:50%;border:2px solid white;background:var(--ion-color-primary);color:#fff;display:flex;align-items:center;justify-content:center;cursor:pointer;transition:transform .2s ease,background-color .2s ease;box-shadow:0 2px 8px #00000026}.edit-button ion-icon{font-size:calc(var(--edit-button-size) * .5)}.edit-button:hover{transform:scale(1.1);background:var(--ion-color-primary-shade)}.edit-button:active{transform:scale(.95)}.edit-button:focus-visible{outline:2px solid var(--ion-color-primary);outline-offset:2px}.loading-overlay{position:absolute;top:0;left:0;width:100%;height:100%;border-radius:50%;background:#00000080;display:flex;align-items:center;justify-content:center}.loading-overlay ion-spinner{--color: white;width:calc(var(--avatar-size) * .4);height:calc(var(--avatar-size) * .4)}.avatar-upload--loading .edit-button{display:none}.avatar-upload--loading .avatar-image,.avatar-upload--loading .avatar-initials{filter:brightness(.7)}@container (max-width: 60px){.edit-button{--edit-button-size: 24px;--edit-button-offset: 0}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: IonModal, selector: "ion-modal" }, { kind: "component", type: ImageCropComponent, selector: "val-image-crop", inputs: ["image", "aspectRatio", "roundCropper", "resizeToWidth", "i18nNamespace"], outputs: ["cropComplete", "cancel", "loadFailed"] }] }); }
256
- }
257
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AvatarUploadComponent, decorators: [{
258
- type: Component,
259
- args: [{ selector: 'val-avatar-upload', standalone: true, imports: [CommonModule, IonIcon, IonSpinner, IonModal, ImageCropComponent], template: `
260
- <div
261
- class="avatar-upload"
262
- [style.--avatar-size.px]="config().size"
263
- [class.avatar-upload--loading]="loading()"
264
- >
265
- <div class="avatar-container">
266
- <!-- Avatar Image or Initials -->
267
- @if (displayUrl()) {
268
- <img
269
- class="avatar-image"
270
- [src]="displayUrl()"
271
- alt="Avatar"
272
- (error)="onImageError()"
273
- />
274
- } @else {
275
- <div
276
- class="avatar-initials"
277
- [style.background-color]="config().backgroundColor"
278
- >
279
- {{ config().initials || '?' }}
280
- </div>
281
- }
282
-
283
- <!-- Edit Button -->
284
- @if (config().editable && !loading()) {
285
- <button
286
- class="edit-button"
287
- type="button"
288
- (click)="openFilePicker()"
289
- [attr.aria-label]="editButtonLabel()"
290
- >
291
- <ion-icon name="camera-outline"></ion-icon>
292
- </button>
293
- }
294
-
295
- <!-- Loading Overlay -->
296
- @if (loading()) {
297
- <div class="loading-overlay">
298
- <ion-spinner name="crescent"></ion-spinner>
299
- </div>
300
- }
301
- </div>
302
-
303
- <!-- Hidden File Input -->
304
- <input
305
- #fileInput
306
- type="file"
307
- accept="image/jpeg,image/png,image/webp"
308
- (change)="onFileSelected($event)"
309
- hidden
310
- />
311
-
312
- <!-- Crop Modal -->
313
- <ion-modal
314
- [isOpen]="showCropModal()"
315
- (didDismiss)="onCropCancel()"
316
- [breakpoints]="[0, 1]"
317
- [initialBreakpoint]="1"
318
- >
319
- <ng-template>
320
- @if (selectedFile()) {
321
- <val-image-crop
322
- [image]="selectedFile()!"
323
- [aspectRatio]="1"
324
- [roundCropper]="true"
325
- [i18nNamespace]="config().i18nNamespace"
326
- (cropComplete)="onCropComplete($event)"
327
- (cancel)="onCropCancel()"
328
- (loadFailed)="onCropLoadFailed()"
329
- />
330
- }
331
- </ng-template>
332
- </ion-modal>
333
- </div>
334
- `, styles: [".avatar-upload{--avatar-size: 100px;--edit-button-size: 32px;--edit-button-offset: 4px;display:inline-block}.avatar-container{position:relative;width:var(--avatar-size);height:var(--avatar-size);border-radius:50%;overflow:visible}.avatar-image{width:100%;height:100%;border-radius:50%;object-fit:cover;background-color:var(--ion-color-light)}.avatar-initials{width:100%;height:100%;border-radius:50%;display:flex;align-items:center;justify-content:center;font-size:calc(var(--avatar-size) * .4);font-weight:600;color:#fff;text-transform:uppercase;-webkit-user-select:none;user-select:none}.edit-button{position:absolute;bottom:var(--edit-button-offset);right:var(--edit-button-offset);width:var(--edit-button-size);height:var(--edit-button-size);border-radius:50%;border:2px solid white;background:var(--ion-color-primary);color:#fff;display:flex;align-items:center;justify-content:center;cursor:pointer;transition:transform .2s ease,background-color .2s ease;box-shadow:0 2px 8px #00000026}.edit-button ion-icon{font-size:calc(var(--edit-button-size) * .5)}.edit-button:hover{transform:scale(1.1);background:var(--ion-color-primary-shade)}.edit-button:active{transform:scale(.95)}.edit-button:focus-visible{outline:2px solid var(--ion-color-primary);outline-offset:2px}.loading-overlay{position:absolute;top:0;left:0;width:100%;height:100%;border-radius:50%;background:#00000080;display:flex;align-items:center;justify-content:center}.loading-overlay ion-spinner{--color: white;width:calc(var(--avatar-size) * .4);height:calc(var(--avatar-size) * .4)}.avatar-upload--loading .edit-button{display:none}.avatar-upload--loading .avatar-image,.avatar-upload--loading .avatar-initials{filter:brightness(.7)}@container (max-width: 60px){.edit-button{--edit-button-size: 24px;--edit-button-offset: 0}}\n"] }]
335
- }], propDecorators: { fileInput: [{
336
- type: ViewChild,
337
- args: ['fileInput']
338
- }], uploaded: [{
339
- type: Output
340
- }], error: [{
341
- type: Output
342
- }], uploadStart: [{
343
- type: Output
344
- }] } });
345
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar-upload.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/avatar-upload/avatar-upload.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  computed,\n  ElementRef,\n  EventEmitter,\n  inject,\n  input,\n  Output,\n  signal,\n  ViewChild,\n} from '@angular/core';\nimport { IonIcon, IonModal, IonSpinner } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { cameraOutline } from 'ionicons/icons';\nimport { firstValueFrom } from 'rxjs';\nimport { AuthService } from '../../../services/auth';\nimport { StorageService } from '../../../services/firebase';\nimport { I18nService } from '../../../services/i18n';\nimport { ImageService } from '../../../services/image';\nimport { ImageCropComponent } from '../../molecules/image-crop';\nimport {\n  AVATAR_UPLOAD_DEFAULTS,\n  AvatarUploadError,\n  AvatarUploadMetadata,\n  AvatarUploadResult,\n} from './types';\n\naddIcons({ cameraOutline });\n\n/**\n * AvatarUploadComponent\n *\n * A complete avatar upload solution with:\n * - Image selection from device\n * - Crop modal with round preview\n * - Automatic compression and thumbnail generation\n * - Upload to Firebase Storage\n * - Backend sync via AuthService\n *\n * @example Basic usage\n * ```html\n * <val-avatar-upload\n *   [props]=\"{\n *     currentUrl: user()?.avatarUrl,\n *     initials: 'JD',\n *     size: 120\n *   }\"\n *   (uploaded)=\"onAvatarUploaded($event)\"\n *   (error)=\"onError($event)\"\n * />\n * ```\n */\n@Component({\n  selector: 'val-avatar-upload',\n  standalone: true,\n  imports: [CommonModule, IonIcon, IonSpinner, IonModal, ImageCropComponent],\n  template: `\n    <div\n      class=\"avatar-upload\"\n      [style.--avatar-size.px]=\"config().size\"\n      [class.avatar-upload--loading]=\"loading()\"\n    >\n      <div class=\"avatar-container\">\n        <!-- Avatar Image or Initials -->\n        @if (displayUrl()) {\n          <img\n            class=\"avatar-image\"\n            [src]=\"displayUrl()\"\n            alt=\"Avatar\"\n            (error)=\"onImageError()\"\n          />\n        } @else {\n          <div\n            class=\"avatar-initials\"\n            [style.background-color]=\"config().backgroundColor\"\n          >\n            {{ config().initials || '?' }}\n          </div>\n        }\n\n        <!-- Edit Button -->\n        @if (config().editable && !loading()) {\n          <button\n            class=\"edit-button\"\n            type=\"button\"\n            (click)=\"openFilePicker()\"\n            [attr.aria-label]=\"editButtonLabel()\"\n          >\n            <ion-icon name=\"camera-outline\"></ion-icon>\n          </button>\n        }\n\n        <!-- Loading Overlay -->\n        @if (loading()) {\n          <div class=\"loading-overlay\">\n            <ion-spinner name=\"crescent\"></ion-spinner>\n          </div>\n        }\n      </div>\n\n      <!-- Hidden File Input -->\n      <input\n        #fileInput\n        type=\"file\"\n        accept=\"image/jpeg,image/png,image/webp\"\n        (change)=\"onFileSelected($event)\"\n        hidden\n      />\n\n      <!-- Crop Modal -->\n      <ion-modal\n        [isOpen]=\"showCropModal()\"\n        (didDismiss)=\"onCropCancel()\"\n        [breakpoints]=\"[0, 1]\"\n        [initialBreakpoint]=\"1\"\n      >\n        <ng-template>\n          @if (selectedFile()) {\n            <val-image-crop\n              [image]=\"selectedFile()!\"\n              [aspectRatio]=\"1\"\n              [roundCropper]=\"true\"\n              [i18nNamespace]=\"config().i18nNamespace\"\n              (cropComplete)=\"onCropComplete($event)\"\n              (cancel)=\"onCropCancel()\"\n              (loadFailed)=\"onCropLoadFailed()\"\n            />\n          }\n        </ng-template>\n      </ion-modal>\n    </div>\n  `,\n  styleUrls: ['./avatar-upload.component.scss'],\n})\nexport class AvatarUploadComponent {\n  private imageService = inject(ImageService);\n  private storageService = inject(StorageService);\n  private authService = inject(AuthService);\n  private i18n = inject(I18nService);\n\n  @ViewChild('fileInput') fileInput!: ElementRef<HTMLInputElement>;\n\n  /** Component configuration */\n  readonly props = input<AvatarUploadMetadata>({});\n\n  /** Emitted after successful upload and backend sync */\n  @Output() uploaded = new EventEmitter<AvatarUploadResult>();\n\n  /** Emitted on any error during the process */\n  @Output() error = new EventEmitter<AvatarUploadError>();\n\n  /** Emitted when upload starts */\n  @Output() uploadStart = new EventEmitter<void>();\n\n  // Internal state\n  protected loading = signal(false);\n  protected showCropModal = signal(false);\n  protected selectedFile = signal<File | null>(null);\n  protected previewUrl = signal<string | null>(null);\n  protected imageLoadError = signal(false);\n\n  /** Merged config with defaults */\n  protected config = computed(() => ({\n    ...AVATAR_UPLOAD_DEFAULTS,\n    ...this.props(),\n  }));\n\n  /** URL to display (preview takes priority over current) */\n  protected displayUrl = computed(() => {\n    if (this.imageLoadError()) return null;\n    return this.previewUrl() || this.config().currentUrl || null;\n  });\n\n  /** Aria label for edit button */\n  protected editButtonLabel = computed(() => {\n    this.i18n.lang();\n    return this.i18n.t('changePhoto', this.config().i18nNamespace) || 'Cambiar foto';\n  });\n\n  /** Open file picker dialog */\n  openFilePicker(): void {\n    this.fileInput.nativeElement.click();\n  }\n\n  /** Handle file selection */\n  onFileSelected(event: Event): void {\n    const input = event.target as HTMLInputElement;\n    const file = input.files?.[0];\n\n    if (!file) return;\n\n    // Reset input for same file selection\n    input.value = '';\n\n    // Validate file\n    const validation = this.imageService.validate(file, {\n      maxSize: this.config().maxFileSize,\n      allowedTypes: ['image/jpeg', 'image/png', 'image/webp'],\n    });\n\n    if (!validation.valid) {\n      this.emitError(validation.error as AvatarUploadError['type'], validation.message!);\n      return;\n    }\n\n    // Open crop modal\n    this.selectedFile.set(file);\n    this.showCropModal.set(true);\n  }\n\n  /** Handle crop completion */\n  async onCropComplete(croppedBlob: Blob): Promise<void> {\n    this.showCropModal.set(false);\n    this.selectedFile.set(null);\n\n    await this.processAndUpload(croppedBlob);\n  }\n\n  /** Handle crop cancel */\n  onCropCancel(): void {\n    this.showCropModal.set(false);\n    this.selectedFile.set(null);\n  }\n\n  /** Handle crop load failure */\n  onCropLoadFailed(): void {\n    this.showCropModal.set(false);\n    this.selectedFile.set(null);\n    this.emitError('invalidType', this.i18n.t('loadFailed', this.config().i18nNamespace) || 'No se pudo cargar la imagen');\n  }\n\n  /** Handle image load error */\n  onImageError(): void {\n    this.imageLoadError.set(true);\n  }\n\n  /** Process cropped image and upload */\n  private async processAndUpload(croppedBlob: Blob): Promise<void> {\n    this.loading.set(true);\n    this.uploadStart.emit();\n\n    try {\n      const config = this.config();\n\n      // 1. Compress image\n      const compressed = await this.imageService.compress(croppedBlob, {\n        maxWidth: config.maxWidth,\n        maxHeight: config.maxWidth,\n        quality: config.compressQuality,\n      });\n\n      // 2. Generate thumbnail\n      const thumbnail = await this.imageService.thumbnail(\n        compressed.blob,\n        config.thumbnailSize\n      );\n\n      // 3. Set preview immediately\n      this.previewUrl.set(compressed.dataUrl);\n      this.imageLoadError.set(false);\n\n      // 4. Get user ID for storage path\n      const userId = this.authService.user()?.userId;\n      if (!userId) {\n        throw new Error('User not authenticated');\n      }\n\n      // 5. Upload to Firebase Storage\n      const timestamp = Date.now();\n      const avatarPath = `${config.storagePath}/${userId}/avatar_${timestamp}.jpg`;\n      const thumbPath = `${config.storagePath}/${userId}/thumb_${timestamp}.jpg`;\n\n      const [avatarResult, thumbResult] = await Promise.all([\n        this.storageService.uploadAndGetUrl(avatarPath, compressed.blob),\n        this.storageService.uploadAndGetUrl(thumbPath, thumbnail.blob),\n      ]);\n\n      // 6. Update backend\n      await firstValueFrom(\n        this.authService.updateAvatar({\n          avatarUrl: avatarResult.downloadUrl,\n          avatarThumbnail: thumbResult.downloadUrl,\n        })\n      );\n\n      // 7. Emit success\n      const result: AvatarUploadResult = {\n        avatarUrl: avatarResult.downloadUrl,\n        thumbnailUrl: thumbResult.downloadUrl,\n      };\n\n      this.uploaded.emit(result);\n    } catch (err) {\n      // Revert preview on error\n      this.previewUrl.set(null);\n\n      const message =\n        err instanceof Error\n          ? err.message\n          : this.i18n.t('uploadError', this.config().i18nNamespace) || 'Error al subir la imagen';\n\n      this.emitError('uploadFailed', message, err);\n    } finally {\n      this.loading.set(false);\n    }\n  }\n\n  /** Emit error event */\n  private emitError(\n    type: AvatarUploadError['type'],\n    message: string,\n    originalError?: unknown\n  ): void {\n    this.error.emit({ type, message, originalError });\n  }\n}\n"]}
@@ -1,15 +0,0 @@
1
- /**
2
- * Default values
3
- */
4
- export const AVATAR_UPLOAD_DEFAULTS = {
5
- size: 100,
6
- editable: true,
7
- storagePath: 'avatars',
8
- i18nNamespace: 'AvatarUpload',
9
- maxFileSize: 10 * 1024 * 1024, // 10MB
10
- compressQuality: 0.8,
11
- maxWidth: 800,
12
- thumbnailSize: 150,
13
- backgroundColor: '#6366f1', // Indigo
14
- };
15
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL2F2YXRhci11cGxvYWQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBeURBOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sc0JBQXNCLEdBQUc7SUFDcEMsSUFBSSxFQUFFLEdBQUc7SUFDVCxRQUFRLEVBQUUsSUFBSTtJQUNkLFdBQVcsRUFBRSxTQUFTO0lBQ3RCLGFBQWEsRUFBRSxjQUFjO0lBQzdCLFdBQVcsRUFBRSxFQUFFLEdBQUcsSUFBSSxHQUFHLElBQUksRUFBRSxPQUFPO0lBQ3RDLGVBQWUsRUFBRSxHQUFHO0lBQ3BCLFFBQVEsRUFBRSxHQUFHO0lBQ2IsYUFBYSxFQUFFLEdBQUc7SUFDbEIsZUFBZSxFQUFFLFNBQVMsRUFBRSxTQUFTO0NBQ3RDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvbmZpZ3VyYXRpb24gZm9yIEF2YXRhclVwbG9hZENvbXBvbmVudFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEF2YXRhclVwbG9hZE1ldGFkYXRhIHtcbiAgLyoqIEN1cnJlbnQgYXZhdGFyIFVSTCAqL1xuICBjdXJyZW50VXJsPzogc3RyaW5nO1xuICAvKiogSW5pdGlhbHMgdG8gc2hvdyB3aGVuIG5vIGF2YXRhciAoZS5nLiwgXCJKRFwiIGZvciBKb2huIERvZSkgKi9cbiAgaW5pdGlhbHM/OiBzdHJpbmc7XG4gIC8qKiBCYWNrZ3JvdW5kIGNvbG9yIGZvciBpbml0aWFscyBhdmF0YXIgKi9cbiAgYmFja2dyb3VuZENvbG9yPzogc3RyaW5nO1xuICAvKiogQXZhdGFyIHNpemUgaW4gcGl4ZWxzIChkZWZhdWx0OiAxMDApICovXG4gIHNpemU/OiBudW1iZXI7XG4gIC8qKiBTaG93IGVkaXQgYnV0dG9uIChkZWZhdWx0OiB0cnVlKSAqL1xuICBlZGl0YWJsZT86IGJvb2xlYW47XG4gIC8qKiBTdG9yYWdlIHBhdGggcHJlZml4IHdpdGhvdXQgdXNlcklkIChkZWZhdWx0OiAnYXZhdGFycycpICovXG4gIHN0b3JhZ2VQYXRoPzogc3RyaW5nO1xuICAvKiogaTE4biBuYW1lc3BhY2UgZm9yIGxhYmVscyAoZGVmYXVsdDogJ0F2YXRhclVwbG9hZCcpICovXG4gIGkxOG5OYW1lc3BhY2U/OiBzdHJpbmc7XG4gIC8qKiBNYXggZmlsZSBzaXplIGluIGJ5dGVzIChkZWZhdWx0OiAxME1CKSAqL1xuICBtYXhGaWxlU2l6ZT86IG51bWJlcjtcbiAgLyoqIFF1YWxpdHkgZm9yIGNvbXByZXNzZWQgaW1hZ2UgMC0xIChkZWZhdWx0OiAwLjgpICovXG4gIGNvbXByZXNzUXVhbGl0eT86IG51bWJlcjtcbiAgLyoqIE1heCB3aWR0aCBmb3IgYXZhdGFyIChkZWZhdWx0OiA4MDApICovXG4gIG1heFdpZHRoPzogbnVtYmVyO1xuICAvKiogVGh1bWJuYWlsIHNpemUgKGRlZmF1bHQ6IDE1MCkgKi9cbiAgdGh1bWJuYWlsU2l6ZT86IG51bWJlcjtcbn1cblxuLyoqXG4gKiBSZXN1bHQgZW1pdHRlZCBhZnRlciBzdWNjZXNzZnVsIHVwbG9hZFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEF2YXRhclVwbG9hZFJlc3VsdCB7XG4gIC8qKiBGdWxsLXNpemUgYXZhdGFyIFVSTCAqL1xuICBhdmF0YXJVcmw6IHN0cmluZztcbiAgLyoqIFRodW1ibmFpbCBVUkwgKi9cbiAgdGh1bWJuYWlsVXJsOiBzdHJpbmc7XG59XG5cbi8qKlxuICogRXJyb3IgdHlwZXMgdGhhdCBjYW4gb2NjdXIgZHVyaW5nIHVwbG9hZFxuICovXG5leHBvcnQgdHlwZSBBdmF0YXJVcGxvYWRFcnJvclR5cGUgPVxuICB8ICdpbnZhbGlkVHlwZSdcbiAgfCAnZmlsZVRvb0xhcmdlJ1xuICB8ICd1cGxvYWRGYWlsZWQnXG4gIHwgJ2JhY2tlbmRGYWlsZWQnXG4gIHwgJ2NhbmNlbGxlZCc7XG5cbi8qKlxuICogRXJyb3Igb2JqZWN0IGVtaXR0ZWQgb24gZmFpbHVyZVxuICovXG5leHBvcnQgaW50ZXJmYWNlIEF2YXRhclVwbG9hZEVycm9yIHtcbiAgdHlwZTogQXZhdGFyVXBsb2FkRXJyb3JUeXBlO1xuICBtZXNzYWdlOiBzdHJpbmc7XG4gIG9yaWdpbmFsRXJyb3I/OiB1bmtub3duO1xufVxuXG4vKipcbiAqIERlZmF1bHQgdmFsdWVzXG4gKi9cbmV4cG9ydCBjb25zdCBBVkFUQVJfVVBMT0FEX0RFRkFVTFRTID0ge1xuICBzaXplOiAxMDAsXG4gIGVkaXRhYmxlOiB0cnVlLFxuICBzdG9yYWdlUGF0aDogJ2F2YXRhcnMnLFxuICBpMThuTmFtZXNwYWNlOiAnQXZhdGFyVXBsb2FkJyxcbiAgbWF4RmlsZVNpemU6IDEwICogMTAyNCAqIDEwMjQsIC8vIDEwTUJcbiAgY29tcHJlc3NRdWFsaXR5OiAwLjgsXG4gIG1heFdpZHRoOiA4MDAsXG4gIHRodW1ibmFpbFNpemU6IDE1MCxcbiAgYmFja2dyb3VuZENvbG9yOiAnIzYzNjZmMScsIC8vIEluZGlnb1xufTtcbiJdfQ==