valtech-components 2.0.677 → 2.0.679
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/components/molecules/image-crop/image-crop.component.mjs +174 -0
- package/esm2022/lib/components/molecules/image-crop/index.mjs +2 -0
- package/esm2022/lib/components/organisms/avatar-upload/avatar-upload.component.mjs +345 -0
- package/esm2022/lib/components/organisms/avatar-upload/types.mjs +15 -0
- package/esm2022/lib/components/organisms/bottom-nav/bottom-nav.component.mjs +3 -3
- package/esm2022/lib/services/auth/auth.service.mjs +10 -1
- package/esm2022/lib/services/auth/types.mjs +1 -1
- package/esm2022/lib/services/image/image.service.mjs +244 -0
- package/esm2022/lib/services/image/index.mjs +3 -0
- package/esm2022/lib/services/image/types.mjs +13 -0
- package/esm2022/lib/version.mjs +2 -2
- package/esm2022/public-api.mjs +7 -1
- package/fesm2022/valtech-components.mjs +784 -5
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/rights-footer/rights-footer.component.d.ts +1 -1
- package/lib/components/atoms/text/text.component.d.ts +1 -1
- package/lib/components/molecules/features-list/features-list.component.d.ts +1 -1
- package/lib/components/molecules/image-crop/image-crop.component.d.ts +59 -0
- package/lib/components/molecules/image-crop/index.d.ts +1 -0
- package/lib/components/organisms/article/article.component.d.ts +4 -4
- package/lib/components/organisms/avatar-upload/avatar-upload.component.d.ts +82 -0
- package/lib/components/organisms/avatar-upload/types.d.ts +62 -0
- package/lib/components/organisms/bottom-nav/bottom-nav.component.d.ts +1 -1
- package/lib/components/organisms/toolbar/toolbar.component.d.ts +1 -1
- package/lib/services/auth/auth.service.d.ts +6 -1
- package/lib/services/auth/types.d.ts +18 -0
- package/lib/services/image/image.service.d.ts +76 -0
- package/lib/services/image/index.d.ts +2 -0
- package/lib/services/image/types.d.ts +74 -0
- package/lib/version.d.ts +1 -1
- package/package.json +6 -2
- package/public-api.d.ts +4 -0
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, computed, EventEmitter, inject, input, Output, signal, } from '@angular/core';
|
|
3
|
+
import { IonButton, IonButtons, IonContent, IonHeader, IonTitle, IonToolbar, } from '@ionic/angular/standalone';
|
|
4
|
+
import { ImageCropperComponent } from 'ngx-image-cropper';
|
|
5
|
+
import { I18nService } from '../../../services/i18n';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
/**
|
|
8
|
+
* ImageCropComponent
|
|
9
|
+
*
|
|
10
|
+
* A modal-ready component for cropping images with a specified aspect ratio.
|
|
11
|
+
* Uses ngx-image-cropper internally and provides a simple interface.
|
|
12
|
+
*
|
|
13
|
+
* @example Inside an ion-modal
|
|
14
|
+
* ```html
|
|
15
|
+
* <ion-modal [isOpen]="showCropModal">
|
|
16
|
+
* <ng-template>
|
|
17
|
+
* <val-image-crop
|
|
18
|
+
* [image]="selectedFile"
|
|
19
|
+
* [aspectRatio]="1"
|
|
20
|
+
* [roundCropper]="true"
|
|
21
|
+
* (cropComplete)="onCropComplete($event)"
|
|
22
|
+
* (cancel)="showCropModal = false"
|
|
23
|
+
* />
|
|
24
|
+
* </ng-template>
|
|
25
|
+
* </ion-modal>
|
|
26
|
+
* ```
|
|
27
|
+
*/
|
|
28
|
+
export class ImageCropComponent {
|
|
29
|
+
constructor() {
|
|
30
|
+
this.i18n = inject(I18nService);
|
|
31
|
+
/** Image file to crop */
|
|
32
|
+
this.image = input.required();
|
|
33
|
+
/** Aspect ratio (1 for square, 16/9 for widescreen, etc.) */
|
|
34
|
+
this.aspectRatio = input(1);
|
|
35
|
+
/** Use round cropper (for avatars) */
|
|
36
|
+
this.roundCropper = input(true);
|
|
37
|
+
/** Resize output to specific width (0 = no resize) */
|
|
38
|
+
this.resizeToWidth = input(0);
|
|
39
|
+
/** i18n namespace for labels */
|
|
40
|
+
this.i18nNamespace = input('ImageCrop');
|
|
41
|
+
/** Emitted when crop is confirmed with the cropped blob */
|
|
42
|
+
this.cropComplete = new EventEmitter();
|
|
43
|
+
/** Emitted when user cancels the crop */
|
|
44
|
+
this.cancel = new EventEmitter();
|
|
45
|
+
/** Emitted when image fails to load */
|
|
46
|
+
this.loadFailed = new EventEmitter();
|
|
47
|
+
/** Internal signal for cropped blob */
|
|
48
|
+
this.croppedBlob = signal(null);
|
|
49
|
+
/** Computed text for cancel button */
|
|
50
|
+
this.cancelText = computed(() => {
|
|
51
|
+
this.i18n.lang(); // Track language changes
|
|
52
|
+
return this.i18n.t('cancel', 'Common') || 'Cancelar';
|
|
53
|
+
});
|
|
54
|
+
/** Computed text for confirm button */
|
|
55
|
+
this.confirmText = computed(() => {
|
|
56
|
+
this.i18n.lang();
|
|
57
|
+
return this.i18n.t('confirm', 'Common') || 'Confirmar';
|
|
58
|
+
});
|
|
59
|
+
/** Computed text for title */
|
|
60
|
+
this.titleText = computed(() => {
|
|
61
|
+
this.i18n.lang();
|
|
62
|
+
return this.i18n.t('cropImage', this.i18nNamespace()) || 'Recortar imagen';
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
/** Handle crop event from ngx-image-cropper */
|
|
66
|
+
onImageCropped(event) {
|
|
67
|
+
if (event.blob) {
|
|
68
|
+
this.croppedBlob.set(event.blob);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
/** Confirm and emit the cropped blob */
|
|
72
|
+
confirmCrop() {
|
|
73
|
+
const blob = this.croppedBlob();
|
|
74
|
+
if (blob) {
|
|
75
|
+
this.cropComplete.emit(blob);
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
/** Handle load failure */
|
|
79
|
+
onLoadFailed() {
|
|
80
|
+
this.loadFailed.emit();
|
|
81
|
+
}
|
|
82
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ImageCropComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
83
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: ImageCropComponent, isStandalone: true, selector: "val-image-crop", inputs: { image: { classPropertyName: "image", publicName: "image", isSignal: true, isRequired: true, transformFunction: null }, aspectRatio: { classPropertyName: "aspectRatio", publicName: "aspectRatio", isSignal: true, isRequired: false, transformFunction: null }, roundCropper: { classPropertyName: "roundCropper", publicName: "roundCropper", isSignal: true, isRequired: false, transformFunction: null }, resizeToWidth: { classPropertyName: "resizeToWidth", publicName: "resizeToWidth", isSignal: true, isRequired: false, transformFunction: null }, i18nNamespace: { classPropertyName: "i18nNamespace", publicName: "i18nNamespace", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { cropComplete: "cropComplete", cancel: "cancel", loadFailed: "loadFailed" }, ngImport: i0, template: `
|
|
84
|
+
<ion-header>
|
|
85
|
+
<ion-toolbar>
|
|
86
|
+
<ion-buttons slot="start">
|
|
87
|
+
<ion-button (click)="cancel.emit()" color="medium">
|
|
88
|
+
{{ cancelText() }}
|
|
89
|
+
</ion-button>
|
|
90
|
+
</ion-buttons>
|
|
91
|
+
<ion-title>{{ titleText() }}</ion-title>
|
|
92
|
+
<ion-buttons slot="end">
|
|
93
|
+
<ion-button
|
|
94
|
+
(click)="confirmCrop()"
|
|
95
|
+
color="primary"
|
|
96
|
+
[strong]="true"
|
|
97
|
+
[disabled]="!croppedBlob()"
|
|
98
|
+
>
|
|
99
|
+
{{ confirmText() }}
|
|
100
|
+
</ion-button>
|
|
101
|
+
</ion-buttons>
|
|
102
|
+
</ion-toolbar>
|
|
103
|
+
</ion-header>
|
|
104
|
+
|
|
105
|
+
<ion-content class="image-crop-content">
|
|
106
|
+
<image-cropper
|
|
107
|
+
[imageFile]="image()"
|
|
108
|
+
[aspectRatio]="aspectRatio()"
|
|
109
|
+
[maintainAspectRatio]="true"
|
|
110
|
+
[roundCropper]="roundCropper()"
|
|
111
|
+
[resizeToWidth]="resizeToWidth()"
|
|
112
|
+
format="jpeg"
|
|
113
|
+
outputType="blob"
|
|
114
|
+
(imageCropped)="onImageCropped($event)"
|
|
115
|
+
(loadImageFailed)="onLoadFailed()"
|
|
116
|
+
/>
|
|
117
|
+
</ion-content>
|
|
118
|
+
`, isInline: true, styles: [":host{display:flex;flex-direction:column;height:100%}.image-crop-content{--background: var(--ion-color-dark)}.image-crop-content::part(scroll){display:flex;flex-direction:column}image-cropper{--cropper-outline-color: rgba(255, 255, 255, .3);--cropper-background-color: var(--ion-color-dark);flex:1;height:100%;max-height:calc(100vh - 56px)}::ng-deep .ngx-ic-component{height:100%!important}::ng-deep .ngx-ic-source-image{max-height:100%!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonHeader, selector: "ion-header", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: IonTitle, selector: "ion-title", inputs: ["color", "size"] }, { kind: "component", type: IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }, { kind: "component", type: ImageCropperComponent, selector: "image-cropper", inputs: ["imageChangedEvent", "imageURL", "imageBase64", "imageFile", "imageAltText", "options", "cropperFrameAriaLabel", "output", "format", "autoCrop", "cropper", "transform", "maintainAspectRatio", "aspectRatio", "resetCropOnAspectRatioChange", "resizeToWidth", "resizeToHeight", "cropperMinWidth", "cropperMinHeight", "cropperMaxHeight", "cropperMaxWidth", "cropperStaticWidth", "cropperStaticHeight", "canvasRotation", "initialStepSize", "roundCropper", "onlyScaleDown", "imageQuality", "backgroundColor", "containWithinAspectRatio", "hideResizeSquares", "allowMoveImage", "checkImageType", "alignImage", "disabled", "hidden"], outputs: ["imageCropped", "startCropImage", "imageLoaded", "cropperReady", "loadImageFailed", "transformChange", "cropperChange"] }] }); }
|
|
119
|
+
}
|
|
120
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ImageCropComponent, decorators: [{
|
|
121
|
+
type: Component,
|
|
122
|
+
args: [{ selector: 'val-image-crop', standalone: true, imports: [
|
|
123
|
+
CommonModule,
|
|
124
|
+
IonHeader,
|
|
125
|
+
IonToolbar,
|
|
126
|
+
IonTitle,
|
|
127
|
+
IonButtons,
|
|
128
|
+
IonButton,
|
|
129
|
+
IonContent,
|
|
130
|
+
ImageCropperComponent,
|
|
131
|
+
], template: `
|
|
132
|
+
<ion-header>
|
|
133
|
+
<ion-toolbar>
|
|
134
|
+
<ion-buttons slot="start">
|
|
135
|
+
<ion-button (click)="cancel.emit()" color="medium">
|
|
136
|
+
{{ cancelText() }}
|
|
137
|
+
</ion-button>
|
|
138
|
+
</ion-buttons>
|
|
139
|
+
<ion-title>{{ titleText() }}</ion-title>
|
|
140
|
+
<ion-buttons slot="end">
|
|
141
|
+
<ion-button
|
|
142
|
+
(click)="confirmCrop()"
|
|
143
|
+
color="primary"
|
|
144
|
+
[strong]="true"
|
|
145
|
+
[disabled]="!croppedBlob()"
|
|
146
|
+
>
|
|
147
|
+
{{ confirmText() }}
|
|
148
|
+
</ion-button>
|
|
149
|
+
</ion-buttons>
|
|
150
|
+
</ion-toolbar>
|
|
151
|
+
</ion-header>
|
|
152
|
+
|
|
153
|
+
<ion-content class="image-crop-content">
|
|
154
|
+
<image-cropper
|
|
155
|
+
[imageFile]="image()"
|
|
156
|
+
[aspectRatio]="aspectRatio()"
|
|
157
|
+
[maintainAspectRatio]="true"
|
|
158
|
+
[roundCropper]="roundCropper()"
|
|
159
|
+
[resizeToWidth]="resizeToWidth()"
|
|
160
|
+
format="jpeg"
|
|
161
|
+
outputType="blob"
|
|
162
|
+
(imageCropped)="onImageCropped($event)"
|
|
163
|
+
(loadImageFailed)="onLoadFailed()"
|
|
164
|
+
/>
|
|
165
|
+
</ion-content>
|
|
166
|
+
`, styles: [":host{display:flex;flex-direction:column;height:100%}.image-crop-content{--background: var(--ion-color-dark)}.image-crop-content::part(scroll){display:flex;flex-direction:column}image-cropper{--cropper-outline-color: rgba(255, 255, 255, .3);--cropper-background-color: var(--ion-color-dark);flex:1;height:100%;max-height:calc(100vh - 56px)}::ng-deep .ngx-ic-component{height:100%!important}::ng-deep .ngx-ic-source-image{max-height:100%!important}\n"] }]
|
|
167
|
+
}], propDecorators: { cropComplete: [{
|
|
168
|
+
type: Output
|
|
169
|
+
}], cancel: [{
|
|
170
|
+
type: Output
|
|
171
|
+
}], loadFailed: [{
|
|
172
|
+
type: Output
|
|
173
|
+
}] } });
|
|
174
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-crop.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/image-crop/image-crop.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  computed,\n  EventEmitter,\n  inject,\n  input,\n  Output,\n  signal,\n} from '@angular/core';\nimport {\n  IonButton,\n  IonButtons,\n  IonContent,\n  IonHeader,\n  IonTitle,\n  IonToolbar,\n} from '@ionic/angular/standalone';\nimport { ImageCroppedEvent, ImageCropperComponent } from 'ngx-image-cropper';\nimport { I18nService } from '../../../services/i18n';\n\n/**\n * ImageCropComponent\n *\n * A modal-ready component for cropping images with a specified aspect ratio.\n * Uses ngx-image-cropper internally and provides a simple interface.\n *\n * @example Inside an ion-modal\n * ```html\n * <ion-modal [isOpen]=\"showCropModal\">\n *   <ng-template>\n *     <val-image-crop\n *       [image]=\"selectedFile\"\n *       [aspectRatio]=\"1\"\n *       [roundCropper]=\"true\"\n *       (cropComplete)=\"onCropComplete($event)\"\n *       (cancel)=\"showCropModal = false\"\n *     />\n *   </ng-template>\n * </ion-modal>\n * ```\n */\n@Component({\n  selector: 'val-image-crop',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonHeader,\n    IonToolbar,\n    IonTitle,\n    IonButtons,\n    IonButton,\n    IonContent,\n    ImageCropperComponent,\n  ],\n  template: `\n    <ion-header>\n      <ion-toolbar>\n        <ion-buttons slot=\"start\">\n          <ion-button (click)=\"cancel.emit()\" color=\"medium\">\n            {{ cancelText() }}\n          </ion-button>\n        </ion-buttons>\n        <ion-title>{{ titleText() }}</ion-title>\n        <ion-buttons slot=\"end\">\n          <ion-button\n            (click)=\"confirmCrop()\"\n            color=\"primary\"\n            [strong]=\"true\"\n            [disabled]=\"!croppedBlob()\"\n          >\n            {{ confirmText() }}\n          </ion-button>\n        </ion-buttons>\n      </ion-toolbar>\n    </ion-header>\n\n    <ion-content class=\"image-crop-content\">\n      <image-cropper\n        [imageFile]=\"image()\"\n        [aspectRatio]=\"aspectRatio()\"\n        [maintainAspectRatio]=\"true\"\n        [roundCropper]=\"roundCropper()\"\n        [resizeToWidth]=\"resizeToWidth()\"\n        format=\"jpeg\"\n        outputType=\"blob\"\n        (imageCropped)=\"onImageCropped($event)\"\n        (loadImageFailed)=\"onLoadFailed()\"\n      />\n    </ion-content>\n  `,\n  styles: [\n    `\n      :host {\n        display: flex;\n        flex-direction: column;\n        height: 100%;\n      }\n\n      .image-crop-content {\n        --background: var(--ion-color-dark);\n      }\n\n      .image-crop-content::part(scroll) {\n        display: flex;\n        flex-direction: column;\n      }\n\n      image-cropper {\n        --cropper-outline-color: rgba(255, 255, 255, 0.3);\n        --cropper-background-color: var(--ion-color-dark);\n        flex: 1;\n        height: 100%;\n        max-height: calc(100vh - 56px);\n      }\n\n      /* Ensure the cropper wrapper takes full height */\n      ::ng-deep .ngx-ic-component {\n        height: 100% !important;\n      }\n\n      ::ng-deep .ngx-ic-source-image {\n        max-height: 100% !important;\n      }\n    `,\n  ],\n})\nexport class ImageCropComponent {\n  private i18n = inject(I18nService);\n\n  /** Image file to crop */\n  readonly image = input.required<File>();\n\n  /** Aspect ratio (1 for square, 16/9 for widescreen, etc.) */\n  readonly aspectRatio = input<number>(1);\n\n  /** Use round cropper (for avatars) */\n  readonly roundCropper = input<boolean>(true);\n\n  /** Resize output to specific width (0 = no resize) */\n  readonly resizeToWidth = input<number>(0);\n\n  /** i18n namespace for labels */\n  readonly i18nNamespace = input<string>('ImageCrop');\n\n  /** Emitted when crop is confirmed with the cropped blob */\n  @Output() cropComplete = new EventEmitter<Blob>();\n\n  /** Emitted when user cancels the crop */\n  @Output() cancel = new EventEmitter<void>();\n\n  /** Emitted when image fails to load */\n  @Output() loadFailed = new EventEmitter<void>();\n\n  /** Internal signal for cropped blob */\n  protected croppedBlob = signal<Blob | null>(null);\n\n  /** Computed text for cancel button */\n  protected cancelText = computed(() => {\n    this.i18n.lang(); // Track language changes\n    return this.i18n.t('cancel', 'Common') || 'Cancelar';\n  });\n\n  /** Computed text for confirm button */\n  protected confirmText = computed(() => {\n    this.i18n.lang();\n    return this.i18n.t('confirm', 'Common') || 'Confirmar';\n  });\n\n  /** Computed text for title */\n  protected titleText = computed(() => {\n    this.i18n.lang();\n    return this.i18n.t('cropImage', this.i18nNamespace()) || 'Recortar imagen';\n  });\n\n  /** Handle crop event from ngx-image-cropper */\n  onImageCropped(event: ImageCroppedEvent): void {\n    if (event.blob) {\n      this.croppedBlob.set(event.blob);\n    }\n  }\n\n  /** Confirm and emit the cropped blob */\n  confirmCrop(): void {\n    const blob = this.croppedBlob();\n    if (blob) {\n      this.cropComplete.emit(blob);\n    }\n  }\n\n  /** Handle load failure */\n  onLoadFailed(): void {\n    this.loadFailed.emit();\n  }\n}\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export * from './image-crop.component';
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2ltYWdlLWNyb3AvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyx3QkFBd0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vaW1hZ2UtY3JvcC5jb21wb25lbnQnO1xuIl19
|
|
@@ -0,0 +1,345 @@
|
|
|
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"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
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==
|