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.
Files changed (32) hide show
  1. package/esm2022/lib/components/molecules/image-crop/image-crop.component.mjs +174 -0
  2. package/esm2022/lib/components/molecules/image-crop/index.mjs +2 -0
  3. package/esm2022/lib/components/organisms/avatar-upload/avatar-upload.component.mjs +345 -0
  4. package/esm2022/lib/components/organisms/avatar-upload/types.mjs +15 -0
  5. package/esm2022/lib/components/organisms/bottom-nav/bottom-nav.component.mjs +3 -3
  6. package/esm2022/lib/services/auth/auth.service.mjs +10 -1
  7. package/esm2022/lib/services/auth/types.mjs +1 -1
  8. package/esm2022/lib/services/image/image.service.mjs +244 -0
  9. package/esm2022/lib/services/image/index.mjs +3 -0
  10. package/esm2022/lib/services/image/types.mjs +13 -0
  11. package/esm2022/lib/version.mjs +2 -2
  12. package/esm2022/public-api.mjs +7 -1
  13. package/fesm2022/valtech-components.mjs +784 -5
  14. package/fesm2022/valtech-components.mjs.map +1 -1
  15. package/lib/components/atoms/rights-footer/rights-footer.component.d.ts +1 -1
  16. package/lib/components/atoms/text/text.component.d.ts +1 -1
  17. package/lib/components/molecules/features-list/features-list.component.d.ts +1 -1
  18. package/lib/components/molecules/image-crop/image-crop.component.d.ts +59 -0
  19. package/lib/components/molecules/image-crop/index.d.ts +1 -0
  20. package/lib/components/organisms/article/article.component.d.ts +4 -4
  21. package/lib/components/organisms/avatar-upload/avatar-upload.component.d.ts +82 -0
  22. package/lib/components/organisms/avatar-upload/types.d.ts +62 -0
  23. package/lib/components/organisms/bottom-nav/bottom-nav.component.d.ts +1 -1
  24. package/lib/components/organisms/toolbar/toolbar.component.d.ts +1 -1
  25. package/lib/services/auth/auth.service.d.ts +6 -1
  26. package/lib/services/auth/types.d.ts +18 -0
  27. package/lib/services/image/image.service.d.ts +76 -0
  28. package/lib/services/image/index.d.ts +2 -0
  29. package/lib/services/image/types.d.ts +74 -0
  30. package/lib/version.d.ts +1 -1
  31. package/package.json +6 -2
  32. 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==