geonetwork-ui 2.4.0-dev.ac57b75b → 2.4.0-dev.c5f4c491

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.
@@ -6,22 +6,15 @@ import { FormControl } from '@angular/forms';
6
6
  import * as i0 from "@angular/core";
7
7
  import * as i1 from "../../../../../../../libs/data-access/gn4/src";
8
8
  import * as i2 from "../../../../../../ui/inputs/src/lib/image-input/image-input.component";
9
- const extractFileNameFromUrl = (url) => {
10
- const pattern = new RegExp(`attachments/([^/?#]+)(?:[/?#]|$)`, 'i');
11
- const match = url.match(pattern);
12
- return match ? match[1] : '';
13
- };
14
9
  export class OverviewUploadComponent {
15
10
  constructor(recordsApiService, cd) {
16
11
  this.recordsApiService = recordsApiService;
17
12
  this.cd = cd;
18
13
  this.overviewChange = new EventEmitter();
19
14
  this.altTextChange = new EventEmitter();
20
- this.resourceAltText = ''; // = ressourceFileName by default
21
- this.resourceFileName = '';
22
15
  this.errorHandle = (error) => {
23
16
  console.error(error);
24
- this.resourceUrl = null;
17
+ this.resourceUrl = '';
25
18
  this.resourceAltText = '';
26
19
  this.resourceFileName = '';
27
20
  this.overviewChange.emit(null);
@@ -32,15 +25,14 @@ export class OverviewUploadComponent {
32
25
  this.recordsApiService.getAllResources(this.metadataUuid).subscribe({
33
26
  next: (resources) => {
34
27
  if (resources && resources.length > 0) {
35
- this.resourceUrl = new URL(resources[0]?.url);
36
- this.resourceAltText =
37
- this.resourceAltText === ''
38
- ? resources[0].filename
39
- : this.resourceAltText;
40
- this.resourceFileName = extractFileNameFromUrl(resources[0]?.url);
28
+ this.resourceUrl = resources[0].url;
29
+ this.resourceFileName = resources[0].filename;
30
+ if (!this.resourceAltText) {
31
+ this.resourceAltText = this.resourceFileName;
32
+ }
41
33
  }
42
34
  else {
43
- this.resourceUrl = null;
35
+ this.resourceUrl = '';
44
36
  this.resourceAltText = '';
45
37
  this.resourceFileName = '';
46
38
  }
@@ -54,7 +46,7 @@ export class OverviewUploadComponent {
54
46
  .putResource(this.metadataUuid, file, 'public')
55
47
  .subscribe({
56
48
  next: (resource) => {
57
- this.resourceUrl = new URL(resource.url);
49
+ this.resourceUrl = resource.url;
58
50
  this.resourceAltText = resource.filename;
59
51
  this.overviewChange.emit({
60
52
  url: new URL(resource.url),
@@ -70,7 +62,7 @@ export class OverviewUploadComponent {
70
62
  .putResourceFromURL(this.metadataUuid, url, 'public')
71
63
  .subscribe({
72
64
  next: (resource) => {
73
- this.resourceUrl = new URL(resource.url);
65
+ this.resourceUrl = resource.url;
74
66
  this.resourceAltText = resource.filename;
75
67
  this.overviewChange.emit({
76
68
  url: new URL(resource.url),
@@ -84,18 +76,19 @@ export class OverviewUploadComponent {
84
76
  handleAltTextChange(newAltText) {
85
77
  this.resourceAltText = newAltText;
86
78
  this.overviewChange.emit({
87
- url: this.resourceUrl,
79
+ url: new URL(this.resourceUrl),
88
80
  description: this.resourceAltText,
89
81
  });
90
82
  this.cd.markForCheck();
91
83
  }
92
84
  handleDelete() {
85
+ //this.formControl.markAsDirty()
93
86
  this.recordsApiService
94
87
  .delResource(this.metadataUuid, this.resourceFileName)
95
88
  .subscribe({
96
89
  next: () => {
97
- this.resourceAltText = null;
98
- this.resourceUrl = null;
90
+ this.resourceAltText = '';
91
+ this.resourceUrl = '';
99
92
  this.overviewChange.emit(null);
100
93
  this.cd.markForCheck();
101
94
  },
@@ -114,18 +107,18 @@ export class OverviewUploadComponent {
114
107
  else {
115
108
  return;
116
109
  }
117
- if (overview.description && overview.description !== '') {
110
+ if (overview.description) {
118
111
  this.resourceAltText = overview.description;
119
112
  this.cd.markForCheck();
120
113
  }
121
114
  }
122
115
  }
123
116
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: OverviewUploadComponent, deps: [{ token: i1.RecordsApiService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
124
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: OverviewUploadComponent, isStandalone: true, selector: "gn-ui-overview-upload", inputs: { metadataUuid: "metadataUuid", formControl: "formControl" }, outputs: { overviewChange: "overviewChange", altTextChange: "altTextChange" }, usesOnChanges: true, ngImport: i0, template: "<gn-ui-image-input\n [maxSizeMB]=\"5\"\n [previewUrl]=\"resourceUrl\"\n [altText]=\"resourceAltText\"\n [formControl]=\"formControl\"\n (fileChange)=\"handleFileChange($event)\"\n (urlChange)=\"handleUrlChange($event)\"\n (altTextChange)=\"handleAltTextChange($event)\"\n (delete)=\"handleDelete()\"\n></gn-ui-image-input>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: UiInputsModule }, { kind: "component", type: i2.ImageInputComponent, selector: "gn-ui-image-input", inputs: ["formControl", "maxSizeMB", "previewUrl", "altText", "uploadProgress", "uploadError"], outputs: ["fileChange", "urlChange", "uploadCancel", "delete", "altTextChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
117
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: OverviewUploadComponent, isStandalone: true, selector: "gn-ui-overview-upload", inputs: { metadataUuid: "metadataUuid", formControl: "formControl" }, outputs: { overviewChange: "overviewChange", altTextChange: "altTextChange" }, usesOnChanges: true, ngImport: i0, template: "<gn-ui-image-input\n [maxSizeMB]=\"5\"\n [previewUrl]=\"resourceUrl\"\n [altText]=\"resourceAltText\"\n (fileChange)=\"handleFileChange($event)\"\n (urlChange)=\"handleUrlChange($event)\"\n (altTextChange)=\"handleAltTextChange($event)\"\n (delete)=\"handleDelete()\"\n></gn-ui-image-input>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: UiInputsModule }, { kind: "component", type: i2.ImageInputComponent, selector: "gn-ui-image-input", inputs: ["maxSizeMB", "previewUrl", "altText", "uploadProgress", "uploadError"], outputs: ["fileChange", "urlChange", "uploadCancel", "delete", "altTextChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
125
118
  }
126
119
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: OverviewUploadComponent, decorators: [{
127
120
  type: Component,
128
- args: [{ selector: 'gn-ui-overview-upload', standalone: true, imports: [CommonModule, UiInputsModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<gn-ui-image-input\n [maxSizeMB]=\"5\"\n [previewUrl]=\"resourceUrl\"\n [altText]=\"resourceAltText\"\n [formControl]=\"formControl\"\n (fileChange)=\"handleFileChange($event)\"\n (urlChange)=\"handleUrlChange($event)\"\n (altTextChange)=\"handleAltTextChange($event)\"\n (delete)=\"handleDelete()\"\n></gn-ui-image-input>\n" }]
121
+ args: [{ selector: 'gn-ui-overview-upload', standalone: true, imports: [CommonModule, UiInputsModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<gn-ui-image-input\n [maxSizeMB]=\"5\"\n [previewUrl]=\"resourceUrl\"\n [altText]=\"resourceAltText\"\n (fileChange)=\"handleFileChange($event)\"\n (urlChange)=\"handleUrlChange($event)\"\n (altTextChange)=\"handleAltTextChange($event)\"\n (delete)=\"handleDelete()\"\n></gn-ui-image-input>\n" }]
129
122
  }], ctorParameters: function () { return [{ type: i1.RecordsApiService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { metadataUuid: [{
130
123
  type: Input
131
124
  }], formControl: [{
@@ -135,4 +128,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
135
128
  }], altTextChange: [{
136
129
  type: Output
137
130
  }] } });
138
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"overview-upload.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.ts","../../../../../../../../../libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core'\nimport { CommonModule } from '@angular/common'\nimport { RecordsApiService } from '../../../../../../../libs/data-access/gn4/src'\nimport { UiInputsModule } from '../../../../../../../libs/ui/inputs/src'\nimport { FormControl } from '@angular/forms'\nimport { GraphicOverview } from '../../../../../../../libs/common/domain/src/lib/model/record'\n\nconst extractFileNameFromUrl = (url: string): string => {\n  const pattern = new RegExp(`attachments/([^/?#]+)(?:[/?#]|$)`, 'i')\n  const match = url.match(pattern)\n  return match ? match[1] : ''\n}\n\n@Component({\n  selector: 'gn-ui-overview-upload',\n  standalone: true,\n  imports: [CommonModule, UiInputsModule],\n  templateUrl: './overview-upload.component.html',\n  styleUrls: ['./overview-upload.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class OverviewUploadComponent implements OnInit, OnChanges {\n  @Input() metadataUuid: string\n  @Input() formControl!: FormControl\n  @Output() overviewChange = new EventEmitter<GraphicOverview | null>()\n  @Output() altTextChange: EventEmitter<string> = new EventEmitter()\n\n  resourceAltText = '' // = ressourceFileName by default\n  resourceFileName = ''\n  resourceUrl: URL\n\n  constructor(\n    private recordsApiService: RecordsApiService,\n    private cd: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.recordsApiService.getAllResources(this.metadataUuid).subscribe({\n      next: (resources) => {\n        if (resources && resources.length > 0) {\n          this.resourceUrl = new URL(resources[0]?.url)\n          this.resourceAltText =\n            this.resourceAltText === ''\n              ? resources[0].filename\n              : this.resourceAltText\n          this.resourceFileName = extractFileNameFromUrl(resources[0]?.url)\n        } else {\n          this.resourceUrl = null\n          this.resourceAltText = ''\n          this.resourceFileName = ''\n        }\n\n        this.cd.markForCheck()\n      },\n      error: this.errorHandle,\n    })\n  }\n\n  handleFileChange(file: File) {\n    this.recordsApiService\n      .putResource(this.metadataUuid, file, 'public')\n      .subscribe({\n        next: (resource) => {\n          this.resourceUrl = new URL(resource.url)\n          this.resourceAltText = resource.filename\n\n          this.overviewChange.emit({\n            url: new URL(resource.url),\n            description: resource.filename,\n          })\n\n          this.cd.markForCheck()\n        },\n        error: this.errorHandle,\n      })\n  }\n\n  handleUrlChange(url: string) {\n    this.recordsApiService\n      .putResourceFromURL(this.metadataUuid, url, 'public')\n      .subscribe({\n        next: (resource) => {\n          this.resourceUrl = new URL(resource.url)\n          this.resourceAltText = resource.filename\n\n          this.overviewChange.emit({\n            url: new URL(resource.url),\n            description: resource.filename,\n          })\n\n          this.cd.markForCheck()\n        },\n        error: this.errorHandle,\n      })\n  }\n\n  handleAltTextChange(newAltText: string) {\n    this.resourceAltText = newAltText\n\n    this.overviewChange.emit({\n      url: this.resourceUrl,\n      description: this.resourceAltText,\n    })\n    this.cd.markForCheck()\n  }\n\n  handleDelete() {\n    this.recordsApiService\n      .delResource(this.metadataUuid, this.resourceFileName)\n      .subscribe({\n        next: () => {\n          this.resourceAltText = null\n          this.resourceUrl = null\n\n          this.overviewChange.emit(null)\n\n          this.cd.markForCheck()\n        },\n        error: this.errorHandle,\n      })\n  }\n\n  private errorHandle = (error: never) => {\n    console.error(error)\n\n    this.resourceUrl = null\n    this.resourceAltText = ''\n    this.resourceFileName = ''\n\n    this.overviewChange.emit(null)\n\n    this.cd.markForCheck()\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const overviewChanges = changes['formControl']\n    if (\n      overviewChanges &&\n      overviewChanges.currentValue !== overviewChanges.previousValue\n    ) {\n      let overview: GraphicOverview\n      if (\n        overviewChanges.currentValue.value &&\n        overviewChanges.currentValue.value.length > 0\n      ) {\n        overview = overviewChanges.currentValue.value[0] as GraphicOverview\n      } else {\n        return\n      }\n      if (overview.description && overview.description !== '') {\n        this.resourceAltText = overview.description\n        this.cd.markForCheck()\n      }\n    }\n  }\n}\n","<gn-ui-image-input\n  [maxSizeMB]=\"5\"\n  [previewUrl]=\"resourceUrl\"\n  [altText]=\"resourceAltText\"\n  [formControl]=\"formControl\"\n  (fileChange)=\"handleFileChange($event)\"\n  (urlChange)=\"handleUrlChange($event)\"\n  (altTextChange)=\"handleAltTextChange($event)\"\n  (delete)=\"handleDelete()\"\n></gn-ui-image-input>\n"]}
131
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"overview-upload.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.ts","../../../../../../../../../libs/feature/editor/src/lib/components/overview-upload/overview-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core'\nimport { CommonModule } from '@angular/common'\nimport { RecordsApiService } from '../../../../../../../libs/data-access/gn4/src'\nimport { UiInputsModule } from '../../../../../../../libs/ui/inputs/src'\nimport { FormControl } from '@angular/forms'\nimport { GraphicOverview } from '../../../../../../../libs/common/domain/src/lib/model/record'\n\n@Component({\n  selector: 'gn-ui-overview-upload',\n  standalone: true,\n  imports: [CommonModule, UiInputsModule],\n  templateUrl: './overview-upload.component.html',\n  styleUrls: ['./overview-upload.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class OverviewUploadComponent implements OnInit, OnChanges {\n  @Input() metadataUuid: string\n  @Input() formControl!: FormControl\n  @Output() overviewChange = new EventEmitter<GraphicOverview | null>()\n  @Output() altTextChange: EventEmitter<string> = new EventEmitter()\n\n  resourceAltText: string\n  resourceFileName: string\n  resourceUrl: string\n\n  constructor(\n    private recordsApiService: RecordsApiService,\n    private cd: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.recordsApiService.getAllResources(this.metadataUuid).subscribe({\n      next: (resources) => {\n        if (resources && resources.length > 0) {\n          this.resourceUrl = resources[0].url\n          this.resourceFileName = resources[0].filename\n          if (!this.resourceAltText) {\n            this.resourceAltText = this.resourceFileName\n          }\n        } else {\n          this.resourceUrl = ''\n          this.resourceAltText = ''\n          this.resourceFileName = ''\n        }\n\n        this.cd.markForCheck()\n      },\n      error: this.errorHandle,\n    })\n  }\n\n  handleFileChange(file: File) {\n    this.recordsApiService\n      .putResource(this.metadataUuid, file, 'public')\n      .subscribe({\n        next: (resource) => {\n          this.resourceUrl = resource.url\n          this.resourceAltText = resource.filename\n\n          this.overviewChange.emit({\n            url: new URL(resource.url),\n            description: resource.filename,\n          })\n\n          this.cd.markForCheck()\n        },\n        error: this.errorHandle,\n      })\n  }\n\n  handleUrlChange(url: string) {\n    this.recordsApiService\n      .putResourceFromURL(this.metadataUuid, url, 'public')\n      .subscribe({\n        next: (resource) => {\n          this.resourceUrl = resource.url\n          this.resourceAltText = resource.filename\n\n          this.overviewChange.emit({\n            url: new URL(resource.url),\n            description: resource.filename,\n          })\n\n          this.cd.markForCheck()\n        },\n        error: this.errorHandle,\n      })\n  }\n\n  handleAltTextChange(newAltText: string) {\n    this.resourceAltText = newAltText\n\n    this.overviewChange.emit({\n      url: new URL(this.resourceUrl),\n      description: this.resourceAltText,\n    })\n\n    this.cd.markForCheck()\n  }\n\n  handleDelete() {\n    //this.formControl.markAsDirty()\n    this.recordsApiService\n      .delResource(this.metadataUuid, this.resourceFileName)\n      .subscribe({\n        next: () => {\n          this.resourceAltText = ''\n          this.resourceUrl = ''\n\n          this.overviewChange.emit(null)\n\n          this.cd.markForCheck()\n        },\n        error: this.errorHandle,\n      })\n  }\n\n  private errorHandle = (error: never) => {\n    console.error(error)\n\n    this.resourceUrl = ''\n    this.resourceAltText = ''\n    this.resourceFileName = ''\n\n    this.overviewChange.emit(null)\n\n    this.cd.markForCheck()\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const overviewChanges = changes['formControl']\n    if (\n      overviewChanges &&\n      overviewChanges.currentValue !== overviewChanges.previousValue\n    ) {\n      let overview: GraphicOverview\n      if (\n        overviewChanges.currentValue.value &&\n        overviewChanges.currentValue.value.length > 0\n      ) {\n        overview = overviewChanges.currentValue.value[0] as GraphicOverview\n      } else {\n        return\n      }\n      if (overview.description) {\n        this.resourceAltText = overview.description\n        this.cd.markForCheck()\n      }\n    }\n  }\n}\n","<gn-ui-image-input\n  [maxSizeMB]=\"5\"\n  [previewUrl]=\"resourceUrl\"\n  [altText]=\"resourceAltText\"\n  (fileChange)=\"handleFileChange($event)\"\n  (urlChange)=\"handleUrlChange($event)\"\n  (altTextChange)=\"handleAltTextChange($event)\"\n  (delete)=\"handleDelete()\"\n></gn-ui-image-input>\n"]}
@@ -15,7 +15,6 @@ export class RecordFormComponent {
15
15
  if (!model) {
16
16
  return;
17
17
  }
18
- console.log(newValue);
19
18
  this.facade.updateRecordField(model, newValue);
20
19
  }
21
20
  fieldTracker(index, field) {
@@ -31,4 +30,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
31
30
  type: Component,
32
31
  args: [{ selector: 'gn-ui-record-form', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, FormFieldComponent, TranslateModule], template: "<ng-container>\n <div class=\"flex flex-col gap-6 p-8\">\n <ng-container\n *ngFor=\"\n let section of facade.currentSections$ | async;\n trackBy: sectionTracker\n \"\n >\n <ng-container *ngIf=\"!section.hidden\">\n <div class=\"flex flex-col gap-8 border p-8 rounded-[8px] shadow\">\n <div\n class=\"flex flex-col gap-2\"\n [ngClass]=\"section.labelKey ? 'mb-4' : 'hidden'\"\n >\n <div\n *ngIf=\"section.labelKey\"\n class=\"text-2xl font-petrona text-secondary\"\n translate\n >\n {{ section.labelKey }}\n </div>\n <div\n *ngIf=\"section.descriptionKey\"\n class=\"text-secondary-lightest\"\n translate\n >\n {{ section.descriptionKey }}\n </div>\n </div>\n <ng-container\n *ngFor=\"\n let field of section.fieldsWithValues;\n trackBy: fieldTracker\n \"\n >\n <ng-container *ngIf=\"!field.config.hidden\">\n <gn-ui-form-field\n [model]=\"field.config.model!\"\n [config]=\"field.config.formFieldConfig\"\n [value]=\"field.value\"\n (valueChange)=\"\n handleFieldValueChange(field.config.model!, $event)\n \"\n ></gn-ui-form-field>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n </div>\n</ng-container>\n" }]
33
32
  }], ctorParameters: function () { return [{ type: i1.EditorFacade }]; } });
34
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjb3JkLWZvcm0uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9mZWF0dXJlL2VkaXRvci9zcmMvbGliL2NvbXBvbmVudHMvcmVjb3JkLWZvcm0vcmVjb3JkLWZvcm0uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9mZWF0dXJlL2VkaXRvci9zcmMvbGliL2NvbXBvbmVudHMvcmVjb3JkLWZvcm0vcmVjb3JkLWZvcm0uY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBQzlDLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFDbEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDRCQUE0QixDQUFBO0FBRXpELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGNBQWMsQ0FBQTtBQUNqRCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUJBQXFCLENBQUE7Ozs7O0FBY3JELE1BQU0sT0FBTyxtQkFBbUI7SUFDOUIsWUFBbUIsTUFBb0I7UUFBcEIsV0FBTSxHQUFOLE1BQU0sQ0FBYztJQUFHLENBQUM7SUFFM0Msc0JBQXNCLENBQUMsS0FBYSxFQUFFLFFBQTBCO1FBQzlELElBQUksQ0FBQyxLQUFLLEVBQUU7WUFDVixPQUFNO1NBQ1A7UUFDRCxPQUFPLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxDQUFBO1FBQ3JCLElBQUksQ0FBQyxNQUFNLENBQUMsaUJBQWlCLENBQUMsS0FBSyxFQUFFLFFBQVEsQ0FBQyxDQUFBO0lBQ2hELENBQUM7SUFFRCxZQUFZLENBQUMsS0FBYSxFQUFFLEtBQTJCO1FBQ3JELE9BQU8sS0FBSyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUE7SUFDM0IsQ0FBQztJQUVELGNBQWMsQ0FBQyxLQUFhLEVBQUUsT0FBZ0M7UUFDNUQsT0FBTyxPQUFPLENBQUMsUUFBUSxDQUFBO0lBQ3pCLENBQUM7K0dBakJVLG1CQUFtQjttR0FBbkIsbUJBQW1CLDZFQ25CaEMsbW1EQW1EQSx5RERsQ1ksWUFBWSxtWkFBRSxrQkFBa0IsNEhBQUUsZUFBZTs7NEZBRWhELG1CQUFtQjtrQkFSL0IsU0FBUzsrQkFDRSxtQkFBbUIsbUJBR1osdUJBQXVCLENBQUMsTUFBTSxjQUNuQyxJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsa0JBQWtCLEVBQUUsZUFBZSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJ1xuaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnXG5pbXBvcnQgeyBFZGl0b3JGYWNhZGUgfSBmcm9tICcuLi8uLi8rc3RhdGUvZWRpdG9yLmZhY2FkZSdcbmltcG9ydCB7IEVkaXRvckZpZWxkVmFsdWUgfSBmcm9tICcuLi8uLi9tb2RlbHMnXG5pbXBvcnQgeyBGb3JtRmllbGRDb21wb25lbnQgfSBmcm9tICcuL2Zvcm0tZmllbGQnXG5pbXBvcnQgeyBUcmFuc2xhdGVNb2R1bGUgfSBmcm9tICdAbmd4LXRyYW5zbGF0ZS9jb3JlJ1xuaW1wb3J0IHtcbiAgRWRpdG9yRmllbGRXaXRoVmFsdWUsXG4gIEVkaXRvclNlY3Rpb25XaXRoVmFsdWVzLFxufSBmcm9tICcuLi8uLi8rc3RhdGUvZWRpdG9yLm1vZGVscydcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnZ24tdWktcmVjb3JkLWZvcm0nLFxuICB0ZW1wbGF0ZVVybDogJy4vcmVjb3JkLWZvcm0uY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9yZWNvcmQtZm9ybS5jb21wb25lbnQuY3NzJ10sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBGb3JtRmllbGRDb21wb25lbnQsIFRyYW5zbGF0ZU1vZHVsZV0sXG59KVxuZXhwb3J0IGNsYXNzIFJlY29yZEZvcm1Db21wb25lbnQge1xuICBjb25zdHJ1Y3RvcihwdWJsaWMgZmFjYWRlOiBFZGl0b3JGYWNhZGUpIHt9XG5cbiAgaGFuZGxlRmllbGRWYWx1ZUNoYW5nZShtb2RlbDogc3RyaW5nLCBuZXdWYWx1ZTogRWRpdG9yRmllbGRWYWx1ZSkge1xuICAgIGlmICghbW9kZWwpIHtcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBjb25zb2xlLmxvZyhuZXdWYWx1ZSlcbiAgICB0aGlzLmZhY2FkZS51cGRhdGVSZWNvcmRGaWVsZChtb2RlbCwgbmV3VmFsdWUpXG4gIH1cblxuICBmaWVsZFRyYWNrZXIoaW5kZXg6IG51bWJlciwgZmllbGQ6IEVkaXRvckZpZWxkV2l0aFZhbHVlKTogYW55IHtcbiAgICByZXR1cm4gZmllbGQuY29uZmlnLm1vZGVsXG4gIH1cblxuICBzZWN0aW9uVHJhY2tlcihpbmRleDogbnVtYmVyLCBzZWN0aW9uOiBFZGl0b3JTZWN0aW9uV2l0aFZhbHVlcyk6IGFueSB7XG4gICAgcmV0dXJuIHNlY3Rpb24ubGFiZWxLZXlcbiAgfVxufVxuIiwiPG5nLWNvbnRhaW5lcj5cbiAgPGRpdiBjbGFzcz1cImZsZXggZmxleC1jb2wgZ2FwLTYgcC04XCI+XG4gICAgPG5nLWNvbnRhaW5lclxuICAgICAgKm5nRm9yPVwiXG4gICAgICAgIGxldCBzZWN0aW9uIG9mIGZhY2FkZS5jdXJyZW50U2VjdGlvbnMkIHwgYXN5bmM7XG4gICAgICAgIHRyYWNrQnk6IHNlY3Rpb25UcmFja2VyXG4gICAgICBcIlxuICAgID5cbiAgICAgIDxuZy1jb250YWluZXIgKm5nSWY9XCIhc2VjdGlvbi5oaWRkZW5cIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cImZsZXggZmxleC1jb2wgZ2FwLTggYm9yZGVyIHAtOCByb3VuZGVkLVs4cHhdIHNoYWRvd1wiPlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIGNsYXNzPVwiZmxleCBmbGV4LWNvbCBnYXAtMlwiXG4gICAgICAgICAgICBbbmdDbGFzc109XCJzZWN0aW9uLmxhYmVsS2V5ID8gJ21iLTQnIDogJ2hpZGRlbidcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgICAgKm5nSWY9XCJzZWN0aW9uLmxhYmVsS2V5XCJcbiAgICAgICAgICAgICAgY2xhc3M9XCJ0ZXh0LTJ4bCBmb250LXBldHJvbmEgdGV4dC1zZWNvbmRhcnlcIlxuICAgICAgICAgICAgICB0cmFuc2xhdGVcbiAgICAgICAgICAgID5cbiAgICAgICAgICAgICAge3sgc2VjdGlvbi5sYWJlbEtleSB9fVxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgICA8ZGl2XG4gICAgICAgICAgICAgICpuZ0lmPVwic2VjdGlvbi5kZXNjcmlwdGlvbktleVwiXG4gICAgICAgICAgICAgIGNsYXNzPVwidGV4dC1zZWNvbmRhcnktbGlnaHRlc3RcIlxuICAgICAgICAgICAgICB0cmFuc2xhdGVcbiAgICAgICAgICAgID5cbiAgICAgICAgICAgICAge3sgc2VjdGlvbi5kZXNjcmlwdGlvbktleSB9fVxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPG5nLWNvbnRhaW5lclxuICAgICAgICAgICAgKm5nRm9yPVwiXG4gICAgICAgICAgICAgIGxldCBmaWVsZCBvZiBzZWN0aW9uLmZpZWxkc1dpdGhWYWx1ZXM7XG4gICAgICAgICAgICAgIHRyYWNrQnk6IGZpZWxkVHJhY2tlclxuICAgICAgICAgICAgXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ0lmPVwiIWZpZWxkLmNvbmZpZy5oaWRkZW5cIj5cbiAgICAgICAgICAgICAgPGduLXVpLWZvcm0tZmllbGRcbiAgICAgICAgICAgICAgICBbbW9kZWxdPVwiZmllbGQuY29uZmlnLm1vZGVsIVwiXG4gICAgICAgICAgICAgICAgW2NvbmZpZ109XCJmaWVsZC5jb25maWcuZm9ybUZpZWxkQ29uZmlnXCJcbiAgICAgICAgICAgICAgICBbdmFsdWVdPVwiZmllbGQudmFsdWVcIlxuICAgICAgICAgICAgICAgICh2YWx1ZUNoYW5nZSk9XCJcbiAgICAgICAgICAgICAgICAgIGhhbmRsZUZpZWxkVmFsdWVDaGFuZ2UoZmllbGQuY29uZmlnLm1vZGVsISwgJGV2ZW50KVxuICAgICAgICAgICAgICAgIFwiXG4gICAgICAgICAgICAgID48L2duLXVpLWZvcm0tZmllbGQ+XG4gICAgICAgICAgICA8L25nLWNvbnRhaW5lcj5cbiAgICAgICAgICA8L25nLWNvbnRhaW5lcj5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L25nLWNvbnRhaW5lcj5cbiAgICA8L25nLWNvbnRhaW5lcj5cbiAgPC9kaXY+XG48L25nLWNvbnRhaW5lcj5cbiJdfQ==
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjb3JkLWZvcm0uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9mZWF0dXJlL2VkaXRvci9zcmMvbGliL2NvbXBvbmVudHMvcmVjb3JkLWZvcm0vcmVjb3JkLWZvcm0uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9mZWF0dXJlL2VkaXRvci9zcmMvbGliL2NvbXBvbmVudHMvcmVjb3JkLWZvcm0vcmVjb3JkLWZvcm0uY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBQzlDLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFDbEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDRCQUE0QixDQUFBO0FBRXpELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGNBQWMsQ0FBQTtBQUNqRCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUJBQXFCLENBQUE7Ozs7O0FBY3JELE1BQU0sT0FBTyxtQkFBbUI7SUFDOUIsWUFBbUIsTUFBb0I7UUFBcEIsV0FBTSxHQUFOLE1BQU0sQ0FBYztJQUFHLENBQUM7SUFFM0Msc0JBQXNCLENBQUMsS0FBYSxFQUFFLFFBQTBCO1FBQzlELElBQUksQ0FBQyxLQUFLLEVBQUU7WUFDVixPQUFNO1NBQ1A7UUFDRCxJQUFJLENBQUMsTUFBTSxDQUFDLGlCQUFpQixDQUFDLEtBQUssRUFBRSxRQUFRLENBQUMsQ0FBQTtJQUNoRCxDQUFDO0lBRUQsWUFBWSxDQUFDLEtBQWEsRUFBRSxLQUEyQjtRQUNyRCxPQUFPLEtBQUssQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFBO0lBQzNCLENBQUM7SUFFRCxjQUFjLENBQUMsS0FBYSxFQUFFLE9BQWdDO1FBQzVELE9BQU8sT0FBTyxDQUFDLFFBQVEsQ0FBQTtJQUN6QixDQUFDOytHQWhCVSxtQkFBbUI7bUdBQW5CLG1CQUFtQiw2RUNuQmhDLG1tREFtREEseUREbENZLFlBQVksbVpBQUUsa0JBQWtCLDRIQUFFLGVBQWU7OzRGQUVoRCxtQkFBbUI7a0JBUi9CLFNBQVM7K0JBQ0UsbUJBQW1CLG1CQUdaLHVCQUF1QixDQUFDLE1BQU0sY0FDbkMsSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLGtCQUFrQixFQUFFLGVBQWUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbidcbmltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQgfSBmcm9tICdAYW5ndWxhci9jb3JlJ1xuaW1wb3J0IHsgRWRpdG9yRmFjYWRlIH0gZnJvbSAnLi4vLi4vK3N0YXRlL2VkaXRvci5mYWNhZGUnXG5pbXBvcnQgeyBFZGl0b3JGaWVsZFZhbHVlIH0gZnJvbSAnLi4vLi4vbW9kZWxzJ1xuaW1wb3J0IHsgRm9ybUZpZWxkQ29tcG9uZW50IH0gZnJvbSAnLi9mb3JtLWZpZWxkJ1xuaW1wb3J0IHsgVHJhbnNsYXRlTW9kdWxlIH0gZnJvbSAnQG5neC10cmFuc2xhdGUvY29yZSdcbmltcG9ydCB7XG4gIEVkaXRvckZpZWxkV2l0aFZhbHVlLFxuICBFZGl0b3JTZWN0aW9uV2l0aFZhbHVlcyxcbn0gZnJvbSAnLi4vLi4vK3N0YXRlL2VkaXRvci5tb2RlbHMnXG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2duLXVpLXJlY29yZC1mb3JtJyxcbiAgdGVtcGxhdGVVcmw6ICcuL3JlY29yZC1mb3JtLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vcmVjb3JkLWZvcm0uY29tcG9uZW50LmNzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgRm9ybUZpZWxkQ29tcG9uZW50LCBUcmFuc2xhdGVNb2R1bGVdLFxufSlcbmV4cG9ydCBjbGFzcyBSZWNvcmRGb3JtQ29tcG9uZW50IHtcbiAgY29uc3RydWN0b3IocHVibGljIGZhY2FkZTogRWRpdG9yRmFjYWRlKSB7fVxuXG4gIGhhbmRsZUZpZWxkVmFsdWVDaGFuZ2UobW9kZWw6IHN0cmluZywgbmV3VmFsdWU6IEVkaXRvckZpZWxkVmFsdWUpIHtcbiAgICBpZiAoIW1vZGVsKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgdGhpcy5mYWNhZGUudXBkYXRlUmVjb3JkRmllbGQobW9kZWwsIG5ld1ZhbHVlKVxuICB9XG5cbiAgZmllbGRUcmFja2VyKGluZGV4OiBudW1iZXIsIGZpZWxkOiBFZGl0b3JGaWVsZFdpdGhWYWx1ZSk6IGFueSB7XG4gICAgcmV0dXJuIGZpZWxkLmNvbmZpZy5tb2RlbFxuICB9XG5cbiAgc2VjdGlvblRyYWNrZXIoaW5kZXg6IG51bWJlciwgc2VjdGlvbjogRWRpdG9yU2VjdGlvbldpdGhWYWx1ZXMpOiBhbnkge1xuICAgIHJldHVybiBzZWN0aW9uLmxhYmVsS2V5XG4gIH1cbn1cbiIsIjxuZy1jb250YWluZXI+XG4gIDxkaXYgY2xhc3M9XCJmbGV4IGZsZXgtY29sIGdhcC02IHAtOFwiPlxuICAgIDxuZy1jb250YWluZXJcbiAgICAgICpuZ0Zvcj1cIlxuICAgICAgICBsZXQgc2VjdGlvbiBvZiBmYWNhZGUuY3VycmVudFNlY3Rpb25zJCB8IGFzeW5jO1xuICAgICAgICB0cmFja0J5OiBzZWN0aW9uVHJhY2tlclxuICAgICAgXCJcbiAgICA+XG4gICAgICA8bmctY29udGFpbmVyICpuZ0lmPVwiIXNlY3Rpb24uaGlkZGVuXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJmbGV4IGZsZXgtY29sIGdhcC04IGJvcmRlciBwLTggcm91bmRlZC1bOHB4XSBzaGFkb3dcIj5cbiAgICAgICAgICA8ZGl2XG4gICAgICAgICAgICBjbGFzcz1cImZsZXggZmxleC1jb2wgZ2FwLTJcIlxuICAgICAgICAgICAgW25nQ2xhc3NdPVwic2VjdGlvbi5sYWJlbEtleSA/ICdtYi00JyA6ICdoaWRkZW4nXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICA8ZGl2XG4gICAgICAgICAgICAgICpuZ0lmPVwic2VjdGlvbi5sYWJlbEtleVwiXG4gICAgICAgICAgICAgIGNsYXNzPVwidGV4dC0yeGwgZm9udC1wZXRyb25hIHRleHQtc2Vjb25kYXJ5XCJcbiAgICAgICAgICAgICAgdHJhbnNsYXRlXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHt7IHNlY3Rpb24ubGFiZWxLZXkgfX1cbiAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgICAgPGRpdlxuICAgICAgICAgICAgICAqbmdJZj1cInNlY3Rpb24uZGVzY3JpcHRpb25LZXlcIlxuICAgICAgICAgICAgICBjbGFzcz1cInRleHQtc2Vjb25kYXJ5LWxpZ2h0ZXN0XCJcbiAgICAgICAgICAgICAgdHJhbnNsYXRlXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHt7IHNlY3Rpb24uZGVzY3JpcHRpb25LZXkgfX1cbiAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgIDxuZy1jb250YWluZXJcbiAgICAgICAgICAgICpuZ0Zvcj1cIlxuICAgICAgICAgICAgICBsZXQgZmllbGQgb2Ygc2VjdGlvbi5maWVsZHNXaXRoVmFsdWVzO1xuICAgICAgICAgICAgICB0cmFja0J5OiBmaWVsZFRyYWNrZXJcbiAgICAgICAgICAgIFwiXG4gICAgICAgICAgPlxuICAgICAgICAgICAgPG5nLWNvbnRhaW5lciAqbmdJZj1cIiFmaWVsZC5jb25maWcuaGlkZGVuXCI+XG4gICAgICAgICAgICAgIDxnbi11aS1mb3JtLWZpZWxkXG4gICAgICAgICAgICAgICAgW21vZGVsXT1cImZpZWxkLmNvbmZpZy5tb2RlbCFcIlxuICAgICAgICAgICAgICAgIFtjb25maWddPVwiZmllbGQuY29uZmlnLmZvcm1GaWVsZENvbmZpZ1wiXG4gICAgICAgICAgICAgICAgW3ZhbHVlXT1cImZpZWxkLnZhbHVlXCJcbiAgICAgICAgICAgICAgICAodmFsdWVDaGFuZ2UpPVwiXG4gICAgICAgICAgICAgICAgICBoYW5kbGVGaWVsZFZhbHVlQ2hhbmdlKGZpZWxkLmNvbmZpZy5tb2RlbCEsICRldmVudClcbiAgICAgICAgICAgICAgICBcIlxuICAgICAgICAgICAgICA+PC9nbi11aS1mb3JtLWZpZWxkPlxuICAgICAgICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgPC9uZy1jb250YWluZXI+XG4gIDwvZGl2PlxuPC9uZy1jb250YWluZXI+XG4iXX0=
@@ -9,7 +9,6 @@ import { ButtonComponent } from '../button/button.component';
9
9
  import { FilesDropDirective } from '../files-drop/files-drop.directive';
10
10
  import { TranslateModule } from '@ngx-translate/core';
11
11
  import { marker } from '@biesbjerg/ngx-translate-extract-marker';
12
- import { FormControl, ReactiveFormsModule } from '@angular/forms';
13
12
  import * as i0 from "@angular/core";
14
13
  import * as i1 from "@angular/common/http";
15
14
  import * as i2 from "@angular/common";
@@ -90,8 +89,7 @@ export class ImageInputComponent {
90
89
  const file = new File([blob], name);
91
90
  this.fileChange.emit(file);
92
91
  },
93
- error: (error) => {
94
- console.error(error);
92
+ error: () => {
95
93
  this.downloadError = true;
96
94
  this.cd.markForCheck();
97
95
  this.urlChange.emit(this.urlInputValue);
@@ -127,7 +125,6 @@ export class ImageInputComponent {
127
125
  }
128
126
  }
129
127
  handleDelete() {
130
- this.formControl.markAsDirty();
131
128
  this.delete.emit();
132
129
  }
133
130
  toggleAltTextInput() {
@@ -150,7 +147,7 @@ export class ImageInputComponent {
150
147
  });
151
148
  }
152
149
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ImageInputComponent, deps: [{ token: i1.HttpClient }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
153
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: ImageInputComponent, isStandalone: true, selector: "gn-ui-image-input", inputs: { formControl: "formControl", maxSizeMB: "maxSizeMB", previewUrl: "previewUrl", altText: "altText", uploadProgress: "uploadProgress", uploadError: "uploadError" }, outputs: { fileChange: "fileChange", urlChange: "urlChange", uploadCancel: "uploadCancel", delete: "delete", altTextChange: "altTextChange" }, ngImport: i0, template: "<ng-container *ngIf=\"previewUrl; then withImage; else withoutImage\">\n</ng-container>\n\n<ng-template #withImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <div class=\"flex-1 group relative\">\n <img\n class=\"w-full h-full object-cover border-2 border-gray-300 rounded-lg\"\n [alt]=\"altText\"\n loading=\"lazy\"\n [src]=\"previewUrl\"\n />\n <gn-ui-button\n [extraClass]=\"\n 'bg-gray-200 absolute right-2 bottom-2 invisible group-hover:visible'\n \"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined\">delete</mat-icon>\n </gn-ui-button>\n </div>\n <input\n *ngIf=\"showAltTextInput\"\n type=\"text\"\n class=\"py-3 px-2 border-2 border-gray-300 rounded-lg text-sm font-medium\"\n [placeholder]=\"'input.image.altTextPlaceholder' | translate\"\n [value]=\"altText\"\n (change)=\"handleAltTextChange($event)\"\n />\n <div class=\"flex flex-row gap-2\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">delete</mat-icon>\n {{ 'input.image.delete' | translate }}\n </gn-ui-button>\n <gn-ui-button\n *ngIf=\"!showAltTextInput\"\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"toggleAltTextInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">add</mat-icon>\n {{ 'input.image.displayAltTextInput' | translate }}\n </gn-ui-button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #withoutImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <label\n gnUiFilesDrop\n class=\"block flex-1 border-2 border-dashed border-gray-300 rounded-lg p-6 flex flex-col items-center justify-center gap-4 hover:cursor-pointer\"\n (dragFilesOver)=\"handleDragFilesOver($event)\"\n (dropFiles)=\"handleDropFiles($event)\"\n >\n <div class=\"w-14 h-14 rounded-md bg-gray-200 grid\">\n <mat-icon\n *ngIf=\"!dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >image</mat-icon\n >\n <mat-icon\n *ngIf=\"dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >add_box</mat-icon\n >\n <div *ngIf=\"uploadProgress\">\n <mat-progress-spinner\n class=\"place-self-center\"\n [diameter]=\"56\"\n [mode]=\"'determinate'\"\n [value]=\"uploadProgress\"\n ></mat-progress-spinner>\n <span\n class=\"text-sm font-medium relative inline-block width-[30px] bottom-[40px] left-[15px]\"\n >\n {{ uploadProgress }}%\n </span>\n </div>\n <mat-icon\n *ngIf=\"uploadError\"\n class=\"material-symbols-outlined place-self-center text-rose-500\"\n >broken_image</mat-icon\n >\n </div>\n <div class=\"flex flex-col items-center gap-1\">\n <p class=\"font-medium\">{{ getPrimaryText() | translate }}</p>\n <p\n class=\"text-sm\"\n [class]=\"\n uploadProgress || uploadError\n ? 'font-bold text-blue-500 cursor-pointer'\n : 'font-medium text-gray-500'\n \"\n (click)=\"handleSecondaryTextClick()\"\n >\n {{ getSecondaryText() | translate }}\n </p>\n </div>\n <input\n type=\"file\"\n class=\"hidden\"\n (change)=\"handleFileInput($event)\"\n [disabled]=\"showUrlInput || uploadProgress || uploadError\"\n />\n </label>\n <div *ngIf=\"!showUrlInput\" class=\"flex-none\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"displayUrlInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">link</mat-icon>\n {{ 'input.image.displayUrlInput' | translate }}\n </gn-ui-button>\n </div>\n <div *ngIf=\"showUrlInput\" class=\"flex-none flex flex-col gap-2\">\n <div class=\"h-2\"></div>\n <div class=\"flex gap-2 items-center\">\n <div class=\"flex-1 flex rounded-lg\">\n <span\n class=\"material-symbols-outlined px-4 inline-flex items-center min-w-fit rounded-s-lg border-2 border-e-0 border-gray-300\"\n >link</span\n >\n <input\n type=\"text\"\n class=\"py-3 ps-1 block w-full border-2 border-s-0 border-e-0 border-gray-300 text-sm font-medium\"\n placeholder=\"https://exemple.com/image.jpg\"\n (change)=\"handleUrlChange($event)\"\n />\n <gn-ui-button\n class=\"px-1 inline-flex items-center min-w-fit rounded-e-lg border-2 border-s-0 border-gray-300 text-white\"\n [extraClass]=\"\n urlInputValue && !downloadError ? 'bg-blue-500' : 'bg-gray-500'\n \"\n [disabled]=\"!urlInputValue || downloadError\"\n (buttonClick)=\"downloadUrl()\"\n >\n <mat-icon class=\"material-symbols-outlined\">arrow_upward</mat-icon>\n </gn-ui-button>\n </div>\n </div>\n </div>\n </div>\n</ng-template>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ButtonComponent, selector: "gn-ui-button", inputs: ["type", "disabled", "extraClass"], outputs: ["buttonClick"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: FilesDropDirective, selector: "[gnUiFilesDrop]", outputs: ["dragFilesOver", "dropFiles"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i4.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: ReactiveFormsModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
150
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: ImageInputComponent, isStandalone: true, selector: "gn-ui-image-input", inputs: { maxSizeMB: "maxSizeMB", previewUrl: "previewUrl", altText: "altText", uploadProgress: "uploadProgress", uploadError: "uploadError" }, outputs: { fileChange: "fileChange", urlChange: "urlChange", uploadCancel: "uploadCancel", delete: "delete", altTextChange: "altTextChange" }, ngImport: i0, template: "<ng-container *ngIf=\"previewUrl; then withImage; else withoutImage\">\n</ng-container>\n\n<ng-template #withImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <div class=\"flex-1 group relative\">\n <img\n class=\"w-full h-full object-cover border-2 border-gray-300 rounded-lg\"\n [alt]=\"altText\"\n loading=\"lazy\"\n [src]=\"previewUrl\"\n />\n <gn-ui-button\n [extraClass]=\"\n 'bg-gray-200 absolute right-2 bottom-2 invisible group-hover:visible'\n \"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined\">delete</mat-icon>\n </gn-ui-button>\n </div>\n <input\n *ngIf=\"showAltTextInput\"\n type=\"text\"\n class=\"py-3 px-2 border-2 border-gray-300 rounded-lg text-sm font-medium\"\n [placeholder]=\"'input.image.altTextPlaceholder' | translate\"\n [value]=\"altText\"\n (change)=\"handleAltTextChange($event)\"\n />\n <div class=\"flex flex-row gap-2\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">delete</mat-icon>\n {{ 'input.image.delete' | translate }}\n </gn-ui-button>\n <gn-ui-button\n *ngIf=\"!showAltTextInput\"\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"toggleAltTextInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">add</mat-icon>\n {{ 'input.image.displayAltTextInput' | translate }}\n </gn-ui-button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #withoutImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <label\n gnUiFilesDrop\n class=\"block flex-1 border-2 border-dashed border-gray-300 rounded-lg p-6 flex flex-col items-center justify-center gap-4\"\n (dragFilesOver)=\"handleDragFilesOver($event)\"\n (dropFiles)=\"handleDropFiles($event)\"\n >\n <div class=\"w-14 h-14 rounded-md bg-gray-200 grid\">\n <mat-icon\n *ngIf=\"!dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >image</mat-icon\n >\n <mat-icon\n *ngIf=\"dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >add_box</mat-icon\n >\n <div *ngIf=\"uploadProgress\">\n <mat-progress-spinner\n class=\"place-self-center\"\n [diameter]=\"56\"\n [mode]=\"'determinate'\"\n [value]=\"uploadProgress\"\n ></mat-progress-spinner>\n <span\n class=\"text-sm font-medium relative inline-block width-[30px] bottom-[40px] left-[15px]\"\n >\n {{ uploadProgress }}%\n </span>\n </div>\n <mat-icon\n *ngIf=\"uploadError\"\n class=\"material-symbols-outlined place-self-center text-rose-500\"\n >broken_image</mat-icon\n >\n </div>\n <div class=\"flex flex-col items-center gap-1\">\n <p class=\"font-medium\">{{ getPrimaryText() | translate }}</p>\n <p\n class=\"text-sm\"\n [class]=\"\n uploadProgress || uploadError\n ? 'font-bold text-blue-500 cursor-pointer'\n : 'font-medium text-gray-500'\n \"\n (click)=\"handleSecondaryTextClick()\"\n >\n {{ getSecondaryText() | translate }}\n </p>\n </div>\n <input\n type=\"file\"\n class=\"hidden\"\n (change)=\"handleFileInput($event)\"\n [disabled]=\"showUrlInput || uploadProgress || uploadError\"\n />\n </label>\n <div *ngIf=\"!showUrlInput\" class=\"flex-none\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"displayUrlInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">link</mat-icon>\n {{ 'input.image.displayUrlInput' | translate }}\n </gn-ui-button>\n </div>\n <div *ngIf=\"showUrlInput\" class=\"flex-none flex flex-col gap-2\">\n <div class=\"h-2\"></div>\n <div class=\"flex gap-2 items-center\">\n <div class=\"flex-1 flex rounded-lg\">\n <span\n class=\"material-symbols-outlined px-4 inline-flex items-center min-w-fit rounded-s-lg border-2 border-e-0 border-gray-300\"\n >link</span\n >\n <input\n type=\"text\"\n class=\"py-3 ps-1 block w-full border-2 border-s-0 border-e-0 border-gray-300 text-sm font-medium\"\n placeholder=\"https://exemple.com/image.jpg\"\n (change)=\"handleUrlChange($event)\"\n />\n <gn-ui-button\n class=\"px-1 inline-flex items-center min-w-fit rounded-e-lg border-2 border-s-0 border-gray-300 text-white\"\n [extraClass]=\"\n urlInputValue && !downloadError ? 'bg-blue-500' : 'bg-gray-500'\n \"\n [disabled]=\"!urlInputValue || downloadError\"\n (buttonClick)=\"downloadUrl()\"\n >\n <mat-icon class=\"material-symbols-outlined\">arrow_upward</mat-icon>\n </gn-ui-button>\n </div>\n </div>\n </div>\n </div>\n</ng-template>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ButtonComponent, selector: "gn-ui-button", inputs: ["type", "disabled", "extraClass"], outputs: ["buttonClick"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: FilesDropDirective, selector: "[gnUiFilesDrop]", outputs: ["dragFilesOver", "dropFiles"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i4.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
154
151
  }
155
152
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ImageInputComponent, decorators: [{
156
153
  type: Component,
@@ -161,11 +158,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
161
158
  FilesDropDirective,
162
159
  MatProgressSpinnerModule,
163
160
  TranslateModule,
164
- ReactiveFormsModule,
165
- ], template: "<ng-container *ngIf=\"previewUrl; then withImage; else withoutImage\">\n</ng-container>\n\n<ng-template #withImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <div class=\"flex-1 group relative\">\n <img\n class=\"w-full h-full object-cover border-2 border-gray-300 rounded-lg\"\n [alt]=\"altText\"\n loading=\"lazy\"\n [src]=\"previewUrl\"\n />\n <gn-ui-button\n [extraClass]=\"\n 'bg-gray-200 absolute right-2 bottom-2 invisible group-hover:visible'\n \"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined\">delete</mat-icon>\n </gn-ui-button>\n </div>\n <input\n *ngIf=\"showAltTextInput\"\n type=\"text\"\n class=\"py-3 px-2 border-2 border-gray-300 rounded-lg text-sm font-medium\"\n [placeholder]=\"'input.image.altTextPlaceholder' | translate\"\n [value]=\"altText\"\n (change)=\"handleAltTextChange($event)\"\n />\n <div class=\"flex flex-row gap-2\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">delete</mat-icon>\n {{ 'input.image.delete' | translate }}\n </gn-ui-button>\n <gn-ui-button\n *ngIf=\"!showAltTextInput\"\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"toggleAltTextInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">add</mat-icon>\n {{ 'input.image.displayAltTextInput' | translate }}\n </gn-ui-button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #withoutImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <label\n gnUiFilesDrop\n class=\"block flex-1 border-2 border-dashed border-gray-300 rounded-lg p-6 flex flex-col items-center justify-center gap-4 hover:cursor-pointer\"\n (dragFilesOver)=\"handleDragFilesOver($event)\"\n (dropFiles)=\"handleDropFiles($event)\"\n >\n <div class=\"w-14 h-14 rounded-md bg-gray-200 grid\">\n <mat-icon\n *ngIf=\"!dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >image</mat-icon\n >\n <mat-icon\n *ngIf=\"dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >add_box</mat-icon\n >\n <div *ngIf=\"uploadProgress\">\n <mat-progress-spinner\n class=\"place-self-center\"\n [diameter]=\"56\"\n [mode]=\"'determinate'\"\n [value]=\"uploadProgress\"\n ></mat-progress-spinner>\n <span\n class=\"text-sm font-medium relative inline-block width-[30px] bottom-[40px] left-[15px]\"\n >\n {{ uploadProgress }}%\n </span>\n </div>\n <mat-icon\n *ngIf=\"uploadError\"\n class=\"material-symbols-outlined place-self-center text-rose-500\"\n >broken_image</mat-icon\n >\n </div>\n <div class=\"flex flex-col items-center gap-1\">\n <p class=\"font-medium\">{{ getPrimaryText() | translate }}</p>\n <p\n class=\"text-sm\"\n [class]=\"\n uploadProgress || uploadError\n ? 'font-bold text-blue-500 cursor-pointer'\n : 'font-medium text-gray-500'\n \"\n (click)=\"handleSecondaryTextClick()\"\n >\n {{ getSecondaryText() | translate }}\n </p>\n </div>\n <input\n type=\"file\"\n class=\"hidden\"\n (change)=\"handleFileInput($event)\"\n [disabled]=\"showUrlInput || uploadProgress || uploadError\"\n />\n </label>\n <div *ngIf=\"!showUrlInput\" class=\"flex-none\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"displayUrlInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">link</mat-icon>\n {{ 'input.image.displayUrlInput' | translate }}\n </gn-ui-button>\n </div>\n <div *ngIf=\"showUrlInput\" class=\"flex-none flex flex-col gap-2\">\n <div class=\"h-2\"></div>\n <div class=\"flex gap-2 items-center\">\n <div class=\"flex-1 flex rounded-lg\">\n <span\n class=\"material-symbols-outlined px-4 inline-flex items-center min-w-fit rounded-s-lg border-2 border-e-0 border-gray-300\"\n >link</span\n >\n <input\n type=\"text\"\n class=\"py-3 ps-1 block w-full border-2 border-s-0 border-e-0 border-gray-300 text-sm font-medium\"\n placeholder=\"https://exemple.com/image.jpg\"\n (change)=\"handleUrlChange($event)\"\n />\n <gn-ui-button\n class=\"px-1 inline-flex items-center min-w-fit rounded-e-lg border-2 border-s-0 border-gray-300 text-white\"\n [extraClass]=\"\n urlInputValue && !downloadError ? 'bg-blue-500' : 'bg-gray-500'\n \"\n [disabled]=\"!urlInputValue || downloadError\"\n (buttonClick)=\"downloadUrl()\"\n >\n <mat-icon class=\"material-symbols-outlined\">arrow_upward</mat-icon>\n </gn-ui-button>\n </div>\n </div>\n </div>\n </div>\n</ng-template>\n" }]
166
- }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { formControl: [{
167
- type: Input
168
- }], maxSizeMB: [{
161
+ ], template: "<ng-container *ngIf=\"previewUrl; then withImage; else withoutImage\">\n</ng-container>\n\n<ng-template #withImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <div class=\"flex-1 group relative\">\n <img\n class=\"w-full h-full object-cover border-2 border-gray-300 rounded-lg\"\n [alt]=\"altText\"\n loading=\"lazy\"\n [src]=\"previewUrl\"\n />\n <gn-ui-button\n [extraClass]=\"\n 'bg-gray-200 absolute right-2 bottom-2 invisible group-hover:visible'\n \"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined\">delete</mat-icon>\n </gn-ui-button>\n </div>\n <input\n *ngIf=\"showAltTextInput\"\n type=\"text\"\n class=\"py-3 px-2 border-2 border-gray-300 rounded-lg text-sm font-medium\"\n [placeholder]=\"'input.image.altTextPlaceholder' | translate\"\n [value]=\"altText\"\n (change)=\"handleAltTextChange($event)\"\n />\n <div class=\"flex flex-row gap-2\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"handleDelete()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">delete</mat-icon>\n {{ 'input.image.delete' | translate }}\n </gn-ui-button>\n <gn-ui-button\n *ngIf=\"!showAltTextInput\"\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"toggleAltTextInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">add</mat-icon>\n {{ 'input.image.displayAltTextInput' | translate }}\n </gn-ui-button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #withoutImage>\n <div class=\"w-full h-full flex flex-col gap-2\">\n <label\n gnUiFilesDrop\n class=\"block flex-1 border-2 border-dashed border-gray-300 rounded-lg p-6 flex flex-col items-center justify-center gap-4\"\n (dragFilesOver)=\"handleDragFilesOver($event)\"\n (dropFiles)=\"handleDropFiles($event)\"\n >\n <div class=\"w-14 h-14 rounded-md bg-gray-200 grid\">\n <mat-icon\n *ngIf=\"!dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >image</mat-icon\n >\n <mat-icon\n *ngIf=\"dragFilesOver && !uploadProgress && !uploadError\"\n class=\"material-symbols-outlined place-self-center text-blue-500\"\n >add_box</mat-icon\n >\n <div *ngIf=\"uploadProgress\">\n <mat-progress-spinner\n class=\"place-self-center\"\n [diameter]=\"56\"\n [mode]=\"'determinate'\"\n [value]=\"uploadProgress\"\n ></mat-progress-spinner>\n <span\n class=\"text-sm font-medium relative inline-block width-[30px] bottom-[40px] left-[15px]\"\n >\n {{ uploadProgress }}%\n </span>\n </div>\n <mat-icon\n *ngIf=\"uploadError\"\n class=\"material-symbols-outlined place-self-center text-rose-500\"\n >broken_image</mat-icon\n >\n </div>\n <div class=\"flex flex-col items-center gap-1\">\n <p class=\"font-medium\">{{ getPrimaryText() | translate }}</p>\n <p\n class=\"text-sm\"\n [class]=\"\n uploadProgress || uploadError\n ? 'font-bold text-blue-500 cursor-pointer'\n : 'font-medium text-gray-500'\n \"\n (click)=\"handleSecondaryTextClick()\"\n >\n {{ getSecondaryText() | translate }}\n </p>\n </div>\n <input\n type=\"file\"\n class=\"hidden\"\n (change)=\"handleFileInput($event)\"\n [disabled]=\"showUrlInput || uploadProgress || uploadError\"\n />\n </label>\n <div *ngIf=\"!showUrlInput\" class=\"flex-none\">\n <gn-ui-button\n [extraClass]=\"'bg-gray-200 font-bold'\"\n (buttonClick)=\"displayUrlInput()\"\n >\n <mat-icon class=\"material-symbols-outlined me-1\">link</mat-icon>\n {{ 'input.image.displayUrlInput' | translate }}\n </gn-ui-button>\n </div>\n <div *ngIf=\"showUrlInput\" class=\"flex-none flex flex-col gap-2\">\n <div class=\"h-2\"></div>\n <div class=\"flex gap-2 items-center\">\n <div class=\"flex-1 flex rounded-lg\">\n <span\n class=\"material-symbols-outlined px-4 inline-flex items-center min-w-fit rounded-s-lg border-2 border-e-0 border-gray-300\"\n >link</span\n >\n <input\n type=\"text\"\n class=\"py-3 ps-1 block w-full border-2 border-s-0 border-e-0 border-gray-300 text-sm font-medium\"\n placeholder=\"https://exemple.com/image.jpg\"\n (change)=\"handleUrlChange($event)\"\n />\n <gn-ui-button\n class=\"px-1 inline-flex items-center min-w-fit rounded-e-lg border-2 border-s-0 border-gray-300 text-white\"\n [extraClass]=\"\n urlInputValue && !downloadError ? 'bg-blue-500' : 'bg-gray-500'\n \"\n [disabled]=\"!urlInputValue || downloadError\"\n (buttonClick)=\"downloadUrl()\"\n >\n <mat-icon class=\"material-symbols-outlined\">arrow_upward</mat-icon>\n </gn-ui-button>\n </div>\n </div>\n </div>\n </div>\n</ng-template>\n" }]
162
+ }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { maxSizeMB: [{
169
163
  type: Input
170
164
  }], previewUrl: [{
171
165
  type: Input
@@ -186,4 +180,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
186
180
  }], altTextChange: [{
187
181
  type: Output
188
182
  }] } });
189
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-input.component.js","sourceRoot":"","sources":["../../../../../../../../libs/ui/inputs/src/lib/image-input/image-input.component.ts","../../../../../../../../libs/ui/inputs/src/lib/image-input/image-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common'\nimport { HttpClient } from '@angular/common/http'\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  Output,\n} from '@angular/core'\nimport { MatIconModule } from '@angular/material/icon'\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner'\nimport { downgradeImage, megabytesToBytes } from '../../../../../../libs/util/shared/src'\nimport { firstValueFrom } from 'rxjs'\nimport { ButtonComponent } from '../button/button.component'\nimport { FilesDropDirective } from '../files-drop/files-drop.directive'\nimport { TranslateModule } from '@ngx-translate/core'\nimport { marker } from '@biesbjerg/ngx-translate-extract-marker'\nimport { FormControl, ReactiveFormsModule } from '@angular/forms'\n\n@Component({\n  selector: 'gn-ui-image-input',\n  templateUrl: './image-input.component.html',\n  styleUrls: ['./image-input.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    CommonModule,\n    ButtonComponent,\n    MatIconModule,\n    FilesDropDirective,\n    MatProgressSpinnerModule,\n    TranslateModule,\n    ReactiveFormsModule,\n  ],\n})\nexport class ImageInputComponent {\n  @Input() formControl!: FormControl\n  @Input() maxSizeMB: number\n  @Input() previewUrl?: URL\n  @Input() altText?: string\n  @Input() uploadProgress?: number\n  @Input() uploadError?: boolean\n  @Output() fileChange: EventEmitter<File> = new EventEmitter()\n  @Output() urlChange: EventEmitter<string> = new EventEmitter()\n  @Output() uploadCancel: EventEmitter<void> = new EventEmitter()\n  @Output() delete: EventEmitter<void> = new EventEmitter()\n  @Output() altTextChange: EventEmitter<string> = new EventEmitter()\n\n  dragFilesOver = false\n  showUrlInput = false\n  downloadError = false\n  showAltTextInput = false\n\n  urlInputValue?: string\n  lastUploadType?: 'file' | 'url'\n  lastUploadContent?: string | File\n\n  constructor(private http: HttpClient, private cd: ChangeDetectorRef) {}\n\n  getPrimaryText() {\n    if (this.uploadError) {\n      return marker('input.image.uploadErrorLabel')\n    }\n    if (this.uploadProgress) {\n      return marker('input.image.uploadProgressLabel')\n    }\n    return marker('input.image.selectFileLabel')\n  }\n\n  getSecondaryText() {\n    if (this.uploadError) {\n      return marker('input.image.uploadErrorRetry')\n    }\n    if (this.uploadProgress) {\n      return marker('input.image.uploadProgressCancel')\n    }\n    return marker('input.image.dropFileLabel')\n  }\n\n  handleDragFilesOver(dragFilesOver: boolean) {\n    if (!this.showUrlInput) {\n      this.dragFilesOver = dragFilesOver\n      this.cd.markForCheck()\n    }\n  }\n\n  handleDropFiles(files: File[]) {\n    if (!this.showUrlInput) {\n      const validFiles = this.filterTypeImage(files)\n      if (validFiles.length > 0) {\n        this.resizeAndEmit(validFiles[0])\n      }\n    }\n  }\n\n  handleFileInput(event: Event) {\n    const inputFiles = Array.from((event.target as HTMLInputElement).files)\n    const validFiles = this.filterTypeImage(inputFiles)\n    if (validFiles.length > 0) {\n      this.resizeAndEmit(validFiles[0])\n    }\n  }\n\n  displayUrlInput() {\n    this.uploadCancel.emit()\n    this.showUrlInput = true\n  }\n\n  handleUrlChange(event: Event) {\n    this.downloadError = false\n    this.urlInputValue = (event.target as HTMLInputElement).value\n  }\n\n  async downloadUrl() {\n    const name = this.urlInputValue.split('/').pop()\n\n    try {\n      const response = await firstValueFrom(\n        this.http.head(this.urlInputValue, { observe: 'response' })\n      )\n      if (\n        response.headers.get('content-type')?.startsWith('image/') &&\n        parseInt(response.headers.get('content-length')) <\n          megabytesToBytes(this.maxSizeMB)\n      ) {\n        this.http.get(this.urlInputValue, { responseType: 'blob' }).subscribe({\n          next: (blob) => {\n            this.cd.markForCheck()\n            const file = new File([blob], name)\n            this.fileChange.emit(file)\n          },\n          error: (error) => {\n            console.error(error)\n            this.downloadError = true\n            this.cd.markForCheck()\n            this.urlChange.emit(this.urlInputValue)\n          },\n        })\n      }\n    } catch {\n      this.downloadError = true\n      this.cd.markForCheck()\n      return\n    }\n  }\n\n  handleSecondaryTextClick() {\n    if (this.uploadError) {\n      this.handleRetry()\n    } else if (this.uploadProgress) {\n      this.handleCancel()\n    }\n  }\n\n  handleCancel() {\n    this.uploadCancel.emit()\n  }\n\n  handleRetry() {\n    switch (this.lastUploadType) {\n      case 'file':\n        this.fileChange.emit(this.lastUploadContent as File)\n        break\n      case 'url':\n        this.urlChange.emit(this.lastUploadContent as string)\n        break\n    }\n  }\n\n  handleDelete() {\n    this.formControl.markAsDirty()\n    this.delete.emit()\n  }\n\n  toggleAltTextInput() {\n    this.showAltTextInput = !this.showAltTextInput\n  }\n\n  handleAltTextChange(event: Event) {\n    const input = event.target as HTMLInputElement\n    this.altTextChange.emit(input.value)\n  }\n\n  private filterTypeImage(files: File[]) {\n    return files.filter((file) => {\n      return file.type.startsWith('image/')\n    })\n  }\n\n  private resizeAndEmit(imageToResize: File) {\n    const maxSizeBytes = megabytesToBytes(this.maxSizeMB)\n    downgradeImage(imageToResize, maxSizeBytes).then((resizedImage) => {\n      const fileToEmit = new File([resizedImage], imageToResize.name)\n      this.fileChange.emit(fileToEmit)\n    })\n  }\n}\n","<ng-container *ngIf=\"previewUrl; then withImage; else withoutImage\">\n</ng-container>\n\n<ng-template #withImage>\n  <div class=\"w-full h-full flex flex-col gap-2\">\n    <div class=\"flex-1 group relative\">\n      <img\n        class=\"w-full h-full object-cover border-2 border-gray-300 rounded-lg\"\n        [alt]=\"altText\"\n        loading=\"lazy\"\n        [src]=\"previewUrl\"\n      />\n      <gn-ui-button\n        [extraClass]=\"\n          'bg-gray-200 absolute right-2 bottom-2 invisible group-hover:visible'\n        \"\n        (buttonClick)=\"handleDelete()\"\n      >\n        <mat-icon class=\"material-symbols-outlined\">delete</mat-icon>\n      </gn-ui-button>\n    </div>\n    <input\n      *ngIf=\"showAltTextInput\"\n      type=\"text\"\n      class=\"py-3 px-2 border-2 border-gray-300 rounded-lg text-sm font-medium\"\n      [placeholder]=\"'input.image.altTextPlaceholder' | translate\"\n      [value]=\"altText\"\n      (change)=\"handleAltTextChange($event)\"\n    />\n    <div class=\"flex flex-row gap-2\">\n      <gn-ui-button\n        [extraClass]=\"'bg-gray-200 font-bold'\"\n        (buttonClick)=\"handleDelete()\"\n      >\n        <mat-icon class=\"material-symbols-outlined me-1\">delete</mat-icon>\n        {{ 'input.image.delete' | translate }}\n      </gn-ui-button>\n      <gn-ui-button\n        *ngIf=\"!showAltTextInput\"\n        [extraClass]=\"'bg-gray-200 font-bold'\"\n        (buttonClick)=\"toggleAltTextInput()\"\n      >\n        <mat-icon class=\"material-symbols-outlined me-1\">add</mat-icon>\n        {{ 'input.image.displayAltTextInput' | translate }}\n      </gn-ui-button>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #withoutImage>\n  <div class=\"w-full h-full flex flex-col gap-2\">\n    <label\n      gnUiFilesDrop\n      class=\"block flex-1 border-2 border-dashed border-gray-300 rounded-lg p-6 flex flex-col items-center justify-center gap-4 hover:cursor-pointer\"\n      (dragFilesOver)=\"handleDragFilesOver($event)\"\n      (dropFiles)=\"handleDropFiles($event)\"\n    >\n      <div class=\"w-14 h-14 rounded-md bg-gray-200 grid\">\n        <mat-icon\n          *ngIf=\"!dragFilesOver && !uploadProgress && !uploadError\"\n          class=\"material-symbols-outlined place-self-center text-blue-500\"\n          >image</mat-icon\n        >\n        <mat-icon\n          *ngIf=\"dragFilesOver && !uploadProgress && !uploadError\"\n          class=\"material-symbols-outlined place-self-center text-blue-500\"\n          >add_box</mat-icon\n        >\n        <div *ngIf=\"uploadProgress\">\n          <mat-progress-spinner\n            class=\"place-self-center\"\n            [diameter]=\"56\"\n            [mode]=\"'determinate'\"\n            [value]=\"uploadProgress\"\n          ></mat-progress-spinner>\n          <span\n            class=\"text-sm font-medium relative inline-block width-[30px] bottom-[40px] left-[15px]\"\n          >\n            {{ uploadProgress }}%\n          </span>\n        </div>\n        <mat-icon\n          *ngIf=\"uploadError\"\n          class=\"material-symbols-outlined place-self-center text-rose-500\"\n          >broken_image</mat-icon\n        >\n      </div>\n      <div class=\"flex flex-col items-center gap-1\">\n        <p class=\"font-medium\">{{ getPrimaryText() | translate }}</p>\n        <p\n          class=\"text-sm\"\n          [class]=\"\n            uploadProgress || uploadError\n              ? 'font-bold text-blue-500 cursor-pointer'\n              : 'font-medium text-gray-500'\n          \"\n          (click)=\"handleSecondaryTextClick()\"\n        >\n          {{ getSecondaryText() | translate }}\n        </p>\n      </div>\n      <input\n        type=\"file\"\n        class=\"hidden\"\n        (change)=\"handleFileInput($event)\"\n        [disabled]=\"showUrlInput || uploadProgress || uploadError\"\n      />\n    </label>\n    <div *ngIf=\"!showUrlInput\" class=\"flex-none\">\n      <gn-ui-button\n        [extraClass]=\"'bg-gray-200 font-bold'\"\n        (buttonClick)=\"displayUrlInput()\"\n      >\n        <mat-icon class=\"material-symbols-outlined me-1\">link</mat-icon>\n        {{ 'input.image.displayUrlInput' | translate }}\n      </gn-ui-button>\n    </div>\n    <div *ngIf=\"showUrlInput\" class=\"flex-none flex flex-col gap-2\">\n      <div class=\"h-2\"></div>\n      <div class=\"flex gap-2 items-center\">\n        <div class=\"flex-1 flex rounded-lg\">\n          <span\n            class=\"material-symbols-outlined px-4 inline-flex items-center min-w-fit rounded-s-lg border-2 border-e-0 border-gray-300\"\n            >link</span\n          >\n          <input\n            type=\"text\"\n            class=\"py-3 ps-1 block w-full border-2 border-s-0 border-e-0 border-gray-300 text-sm font-medium\"\n            placeholder=\"https://exemple.com/image.jpg\"\n            (change)=\"handleUrlChange($event)\"\n          />\n          <gn-ui-button\n            class=\"px-1 inline-flex items-center min-w-fit rounded-e-lg border-2 border-s-0 border-gray-300 text-white\"\n            [extraClass]=\"\n              urlInputValue && !downloadError ? 'bg-blue-500' : 'bg-gray-500'\n            \"\n            [disabled]=\"!urlInputValue || downloadError\"\n            (buttonClick)=\"downloadUrl()\"\n          >\n            <mat-icon class=\"material-symbols-outlined\">arrow_upward</mat-icon>\n          </gn-ui-button>\n        </div>\n      </div>\n    </div>\n  </div>\n</ng-template>\n"]}
183
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-input.component.js","sourceRoot":"","sources":["../../../../../../../../libs/ui/inputs/src/lib/image-input/image-input.component.ts","../../../../../../../../libs/ui/inputs/src/lib/image-input/image-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common'\nimport { HttpClient } from '@angular/common/http'\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  Output,\n} from '@angular/core'\nimport { MatIconModule } from '@angular/material/icon'\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner'\nimport { downgradeImage, megabytesToBytes } from '../../../../../../libs/util/shared/src'\nimport { firstValueFrom } from 'rxjs'\nimport { ButtonComponent } from '../button/button.component'\nimport { FilesDropDirective } from '../files-drop/files-drop.directive'\nimport { TranslateModule } from '@ngx-translate/core'\nimport { marker } from '@biesbjerg/ngx-translate-extract-marker'\n\n@Component({\n  selector: 'gn-ui-image-input',\n  templateUrl: './image-input.component.html',\n  styleUrls: ['./image-input.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [\n    CommonModule,\n    ButtonComponent,\n    MatIconModule,\n    FilesDropDirective,\n    MatProgressSpinnerModule,\n    TranslateModule,\n  ],\n})\nexport class ImageInputComponent {\n  @Input() maxSizeMB: number\n  @Input() previewUrl?: string\n  @Input() altText?: string\n  @Input() uploadProgress?: number\n  @Input() uploadError?: boolean\n  @Output() fileChange: EventEmitter<File> = new EventEmitter()\n  @Output() urlChange: EventEmitter<string> = new EventEmitter()\n  @Output() uploadCancel: EventEmitter<void> = new EventEmitter()\n  @Output() delete: EventEmitter<void> = new EventEmitter()\n  @Output() altTextChange: EventEmitter<string> = new EventEmitter()\n\n  dragFilesOver = false\n  showUrlInput = false\n  downloadError = false\n  showAltTextInput = false\n\n  urlInputValue?: string\n  lastUploadType?: 'file' | 'url'\n  lastUploadContent?: string | File\n\n  constructor(private http: HttpClient, private cd: ChangeDetectorRef) {}\n\n  getPrimaryText() {\n    if (this.uploadError) {\n      return marker('input.image.uploadErrorLabel')\n    }\n    if (this.uploadProgress) {\n      return marker('input.image.uploadProgressLabel')\n    }\n    return marker('input.image.selectFileLabel')\n  }\n\n  getSecondaryText() {\n    if (this.uploadError) {\n      return marker('input.image.uploadErrorRetry')\n    }\n    if (this.uploadProgress) {\n      return marker('input.image.uploadProgressCancel')\n    }\n    return marker('input.image.dropFileLabel')\n  }\n\n  handleDragFilesOver(dragFilesOver: boolean) {\n    if (!this.showUrlInput) {\n      this.dragFilesOver = dragFilesOver\n      this.cd.markForCheck()\n    }\n  }\n\n  handleDropFiles(files: File[]) {\n    if (!this.showUrlInput) {\n      const validFiles = this.filterTypeImage(files)\n      if (validFiles.length > 0) {\n        this.resizeAndEmit(validFiles[0])\n      }\n    }\n  }\n\n  handleFileInput(event: Event) {\n    const inputFiles = Array.from((event.target as HTMLInputElement).files)\n    const validFiles = this.filterTypeImage(inputFiles)\n    if (validFiles.length > 0) {\n      this.resizeAndEmit(validFiles[0])\n    }\n  }\n\n  displayUrlInput() {\n    this.uploadCancel.emit()\n    this.showUrlInput = true\n  }\n\n  handleUrlChange(event: Event) {\n    this.downloadError = false\n    this.urlInputValue = (event.target as HTMLInputElement).value\n  }\n\n  async downloadUrl() {\n    const name = this.urlInputValue.split('/').pop()\n\n    try {\n      const response = await firstValueFrom(\n        this.http.head(this.urlInputValue, { observe: 'response' })\n      )\n      if (\n        response.headers.get('content-type')?.startsWith('image/') &&\n        parseInt(response.headers.get('content-length')) <\n          megabytesToBytes(this.maxSizeMB)\n      ) {\n        this.http.get(this.urlInputValue, { responseType: 'blob' }).subscribe({\n          next: (blob) => {\n            this.cd.markForCheck()\n            const file = new File([blob], name)\n            this.fileChange.emit(file)\n          },\n          error: () => {\n            this.downloadError = true\n            this.cd.markForCheck()\n            this.urlChange.emit(this.urlInputValue)\n          },\n        })\n      }\n    } catch {\n      this.downloadError = true\n      this.cd.markForCheck()\n      return\n    }\n  }\n\n  handleSecondaryTextClick() {\n    if (this.uploadError) {\n      this.handleRetry()\n    } else if (this.uploadProgress) {\n      this.handleCancel()\n    }\n  }\n\n  handleCancel() {\n    this.uploadCancel.emit()\n  }\n\n  handleRetry() {\n    switch (this.lastUploadType) {\n      case 'file':\n        this.fileChange.emit(this.lastUploadContent as File)\n        break\n      case 'url':\n        this.urlChange.emit(this.lastUploadContent as string)\n        break\n    }\n  }\n\n  handleDelete() {\n    this.delete.emit()\n  }\n\n  toggleAltTextInput() {\n    this.showAltTextInput = !this.showAltTextInput\n  }\n\n  handleAltTextChange(event: Event) {\n    const input = event.target as HTMLInputElement\n    this.altTextChange.emit(input.value)\n  }\n\n  private filterTypeImage(files: File[]) {\n    return files.filter((file) => {\n      return file.type.startsWith('image/')\n    })\n  }\n\n  private resizeAndEmit(imageToResize: File) {\n    const maxSizeBytes = megabytesToBytes(this.maxSizeMB)\n    downgradeImage(imageToResize, maxSizeBytes).then((resizedImage) => {\n      const fileToEmit = new File([resizedImage], imageToResize.name)\n      this.fileChange.emit(fileToEmit)\n    })\n  }\n}\n","<ng-container *ngIf=\"previewUrl; then withImage; else withoutImage\">\n</ng-container>\n\n<ng-template #withImage>\n  <div class=\"w-full h-full flex flex-col gap-2\">\n    <div class=\"flex-1 group relative\">\n      <img\n        class=\"w-full h-full object-cover border-2 border-gray-300 rounded-lg\"\n        [alt]=\"altText\"\n        loading=\"lazy\"\n        [src]=\"previewUrl\"\n      />\n      <gn-ui-button\n        [extraClass]=\"\n          'bg-gray-200 absolute right-2 bottom-2 invisible group-hover:visible'\n        \"\n        (buttonClick)=\"handleDelete()\"\n      >\n        <mat-icon class=\"material-symbols-outlined\">delete</mat-icon>\n      </gn-ui-button>\n    </div>\n    <input\n      *ngIf=\"showAltTextInput\"\n      type=\"text\"\n      class=\"py-3 px-2 border-2 border-gray-300 rounded-lg text-sm font-medium\"\n      [placeholder]=\"'input.image.altTextPlaceholder' | translate\"\n      [value]=\"altText\"\n      (change)=\"handleAltTextChange($event)\"\n    />\n    <div class=\"flex flex-row gap-2\">\n      <gn-ui-button\n        [extraClass]=\"'bg-gray-200 font-bold'\"\n        (buttonClick)=\"handleDelete()\"\n      >\n        <mat-icon class=\"material-symbols-outlined me-1\">delete</mat-icon>\n        {{ 'input.image.delete' | translate }}\n      </gn-ui-button>\n      <gn-ui-button\n        *ngIf=\"!showAltTextInput\"\n        [extraClass]=\"'bg-gray-200 font-bold'\"\n        (buttonClick)=\"toggleAltTextInput()\"\n      >\n        <mat-icon class=\"material-symbols-outlined me-1\">add</mat-icon>\n        {{ 'input.image.displayAltTextInput' | translate }}\n      </gn-ui-button>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #withoutImage>\n  <div class=\"w-full h-full flex flex-col gap-2\">\n    <label\n      gnUiFilesDrop\n      class=\"block flex-1 border-2 border-dashed border-gray-300 rounded-lg p-6 flex flex-col items-center justify-center gap-4\"\n      (dragFilesOver)=\"handleDragFilesOver($event)\"\n      (dropFiles)=\"handleDropFiles($event)\"\n    >\n      <div class=\"w-14 h-14 rounded-md bg-gray-200 grid\">\n        <mat-icon\n          *ngIf=\"!dragFilesOver && !uploadProgress && !uploadError\"\n          class=\"material-symbols-outlined place-self-center text-blue-500\"\n          >image</mat-icon\n        >\n        <mat-icon\n          *ngIf=\"dragFilesOver && !uploadProgress && !uploadError\"\n          class=\"material-symbols-outlined place-self-center text-blue-500\"\n          >add_box</mat-icon\n        >\n        <div *ngIf=\"uploadProgress\">\n          <mat-progress-spinner\n            class=\"place-self-center\"\n            [diameter]=\"56\"\n            [mode]=\"'determinate'\"\n            [value]=\"uploadProgress\"\n          ></mat-progress-spinner>\n          <span\n            class=\"text-sm font-medium relative inline-block width-[30px] bottom-[40px] left-[15px]\"\n          >\n            {{ uploadProgress }}%\n          </span>\n        </div>\n        <mat-icon\n          *ngIf=\"uploadError\"\n          class=\"material-symbols-outlined place-self-center text-rose-500\"\n          >broken_image</mat-icon\n        >\n      </div>\n      <div class=\"flex flex-col items-center gap-1\">\n        <p class=\"font-medium\">{{ getPrimaryText() | translate }}</p>\n        <p\n          class=\"text-sm\"\n          [class]=\"\n            uploadProgress || uploadError\n              ? 'font-bold text-blue-500 cursor-pointer'\n              : 'font-medium text-gray-500'\n          \"\n          (click)=\"handleSecondaryTextClick()\"\n        >\n          {{ getSecondaryText() | translate }}\n        </p>\n      </div>\n      <input\n        type=\"file\"\n        class=\"hidden\"\n        (change)=\"handleFileInput($event)\"\n        [disabled]=\"showUrlInput || uploadProgress || uploadError\"\n      />\n    </label>\n    <div *ngIf=\"!showUrlInput\" class=\"flex-none\">\n      <gn-ui-button\n        [extraClass]=\"'bg-gray-200 font-bold'\"\n        (buttonClick)=\"displayUrlInput()\"\n      >\n        <mat-icon class=\"material-symbols-outlined me-1\">link</mat-icon>\n        {{ 'input.image.displayUrlInput' | translate }}\n      </gn-ui-button>\n    </div>\n    <div *ngIf=\"showUrlInput\" class=\"flex-none flex flex-col gap-2\">\n      <div class=\"h-2\"></div>\n      <div class=\"flex gap-2 items-center\">\n        <div class=\"flex-1 flex rounded-lg\">\n          <span\n            class=\"material-symbols-outlined px-4 inline-flex items-center min-w-fit rounded-s-lg border-2 border-e-0 border-gray-300\"\n            >link</span\n          >\n          <input\n            type=\"text\"\n            class=\"py-3 ps-1 block w-full border-2 border-s-0 border-e-0 border-gray-300 text-sm font-medium\"\n            placeholder=\"https://exemple.com/image.jpg\"\n            (change)=\"handleUrlChange($event)\"\n          />\n          <gn-ui-button\n            class=\"px-1 inline-flex items-center min-w-fit rounded-e-lg border-2 border-s-0 border-gray-300 text-white\"\n            [extraClass]=\"\n              urlInputValue && !downloadError ? 'bg-blue-500' : 'bg-gray-500'\n            \"\n            [disabled]=\"!urlInputValue || downloadError\"\n            (buttonClick)=\"downloadUrl()\"\n          >\n            <mat-icon class=\"material-symbols-outlined\">arrow_upward</mat-icon>\n          </gn-ui-button>\n        </div>\n      </div>\n    </div>\n  </div>\n</ng-template>\n"]}