@energycap/components 0.39.3 → 0.39.4-ECAP-23220-bc-file-upload-dialog.20231219-1507

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.
@@ -5,10 +5,12 @@ import { FormControlBase } from '../form-control-base';
5
5
  import * as i0 from "@angular/core";
6
6
  import * as i1 from "../../core/validation-message.service";
7
7
  import * as i2 from "../../shared/form-group.helper";
8
- import * as i3 from "@angular/forms";
9
- import * as i4 from "../button/button.component";
10
- import * as i5 from "../form-control/form-control.component";
11
- import * as i6 from "../form-group/form-group.component";
8
+ import * as i3 from "@angular/common";
9
+ import * as i4 from "@angular/forms";
10
+ import * as i5 from "../button/button.component";
11
+ import * as i6 from "../form-control/form-control.component";
12
+ import * as i7 from "../form-group/form-group.component";
13
+ import * as i8 from "@ngx-translate/core";
12
14
  export const FileTypeExtensions = {
13
15
  zip: ['.zip'],
14
16
  excel: ['.xls', '.xlsx']
@@ -17,8 +19,8 @@ export class FileUploadComponent extends FormControlBase {
17
19
  // static class to create the form group from a parent component
18
20
  static getFormModel(validators, disabled = false) {
19
21
  let formGroup = new UntypedFormGroup({
20
- file: new UntypedFormControl({ value: null, disabled: disabled }, validators),
21
- name: new UntypedFormControl({ value: null, disabled: disabled }, validators),
22
+ files: new UntypedFormControl({ value: null, disabled: disabled }, validators),
23
+ names: new UntypedFormControl({ value: null, disabled: disabled }, validators),
22
24
  base64FileString: new UntypedFormControl(null),
23
25
  uploadResult: new UntypedFormControl(null)
24
26
  });
@@ -39,6 +41,12 @@ export class FileUploadComponent extends FormControlBase {
39
41
  * File output, determines which properties are supplied on the formModel
40
42
  */
41
43
  this.fileOutput = 'base64';
44
+ /**
45
+ * Optional display type that controls whether the file input textbox is displayed or
46
+ * simply a button the user clicks to launch the OS file storage dialog.
47
+ * Default: file
48
+ */
49
+ this.displayType = 'file';
42
50
  }
43
51
  ngOnChanges(changes) {
44
52
  super.ngOnChanges(changes);
@@ -52,35 +60,62 @@ export class FileUploadComponent extends FormControlBase {
52
60
  this.formModel?.get('name')?.valueChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(value => {
53
61
  if (!value) {
54
62
  this.formModel.patchValue({
55
- file: null,
63
+ files: null,
64
+ names: null,
56
65
  base64FileString: null,
57
66
  uploadResult: null
58
67
  });
59
68
  }
60
69
  });
61
70
  }
71
+ // * New fileChange method
62
72
  async fileChange(files) {
63
- let file = files.item(0);
64
- // If there is a file selected and then opened again and click cancel you get null so don't try and set anything
65
- if (file) {
66
- let reader = new FileReader();
67
- reader.onloadend = async (e) => {
68
- let base64FileString = reader?.result?.toString().split(",")[1];
69
- this.processFile(file, base64FileString);
70
- };
71
- if (this.isBase64FileOutput()) {
72
- reader.readAsDataURL(file);
73
- }
74
- else {
75
- await this.processFile(file);
76
- }
77
- // Clear the file inputs value, this will allow the user to pick the same filename and cause
78
- // the fileChange to trigger.
79
- if (this.fileInput) {
80
- this.fileInput.nativeElement.value = '';
73
+ const promises = Array.from(files).map((file) => new Promise((resolve) => {
74
+ if (file) {
75
+ const reader = new FileReader();
76
+ reader.onloadend = async (e) => {
77
+ const base64FileString = reader?.result?.toString().split(",")[1];
78
+ await this.processFile(file, base64FileString);
79
+ resolve(null);
80
+ };
81
+ if (this.isBase64FileOutput()) {
82
+ reader.readAsDataURL(file);
83
+ }
84
+ else {
85
+ resolve(this.processFile(file));
86
+ }
87
+ ;
81
88
  }
89
+ }));
90
+ await Promise.all(promises);
91
+ // Clear the file inputs value, this will allow the user to pick the same filenames again
92
+ // and cause fileChange to re-trigger.
93
+ if (this.fileInput) {
94
+ this.fileInput.nativeElement.value = '';
82
95
  }
83
96
  }
97
+ // * Old Method
98
+ // public async fileChange(files: FileList): Promise<void> {
99
+ // let file = files.item(0);
100
+ // // If there is a file selected and then opened again and click cancel you get null so don't try and set anything
101
+ // if (file) {
102
+ // let reader: FileReader = new FileReader();
103
+ // reader.onloadend = async e => {
104
+ // let base64FileString: string | undefined = reader?.result?.toString().split(",")[1];
105
+ // this.processFile(file!, base64FileString);
106
+ // };
107
+ // if (this.isBase64FileOutput()) {
108
+ // reader.readAsDataURL(file);
109
+ // } else {
110
+ // await this.processFile(file!);
111
+ // }
112
+ // // Clear the file inputs value, this will allow the user to pick the same filename and cause
113
+ // // the fileChange to trigger.
114
+ // if (this.fileInput) {
115
+ // this.fileInput.nativeElement.value = '';
116
+ // }
117
+ // }
118
+ // }
84
119
  /**
85
120
  * Checks the file type and updates the file type accept property. This is what determines the file
86
121
  * type choices that the user will be limited to in the file browse dialog
@@ -133,10 +168,22 @@ export class FileUploadComponent extends FormControlBase {
133
168
  * @param base64FileString
134
169
  * @param onFileSelectedResult
135
170
  */
171
+ // * Old method
136
172
  patchFileResult(file, base64FileString, onFileSelectedResult) {
173
+ // Get the current value of files from formModel
174
+ let files = this.formModel?.get('files')?.value || [];
175
+ let names = this.formModel?.get('names')?.value || [];
176
+ // If the file is not null, append it to the files array
177
+ if (file) {
178
+ files = [...files, file];
179
+ }
180
+ // If the file name is not null, append it to the names array
181
+ if (file?.name) {
182
+ names = [...names, file.name];
183
+ }
137
184
  this.formModel?.patchValue({
138
- file: file,
139
- name: file?.name,
185
+ files: files,
186
+ names: names,
140
187
  base64FileString: base64FileString ?? null
141
188
  });
142
189
  if (onFileSelectedResult) {
@@ -148,10 +195,10 @@ export class FileUploadComponent extends FormControlBase {
148
195
  }
149
196
  }
150
197
  FileUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, deps: [{ token: i1.ValidationMessageService }, { token: i2.FormGroupHelper }], target: i0.ɵɵFactoryTarget.Component });
151
- FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (click)=\"browseBtn.focus()\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\">\r\n <ec-form-control id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: i4.ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: i5.FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: i6.FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }] });
198
+ FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected", displayType: "displayType", buttonLabel: "buttonLabel" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: i5.ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: i6.FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: i7.FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }, { kind: "pipe", type: i8.TranslatePipe, name: "translate" }] });
152
199
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, decorators: [{
153
200
  type: Component,
154
- args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (click)=\"browseBtn.focus()\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\">\r\n <ec-form-control id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
201
+ args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
155
202
  }], ctorParameters: function () { return [{ type: i1.ValidationMessageService }, { type: i2.FormGroupHelper }]; }, propDecorators: { placeholder: [{
156
203
  type: Input
157
204
  }], fileType: [{
@@ -162,8 +209,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
162
209
  type: Input
163
210
  }], onFileSelected: [{
164
211
  type: Input
212
+ }], displayType: [{
213
+ type: Input
214
+ }], buttonLabel: [{
215
+ type: Input
165
216
  }], fileInput: [{
166
217
  type: ViewChild,
167
218
  args: ["fileInput", { read: ElementRef, static: true }]
168
219
  }] } });
169
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.ts","../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn } from '@angular/forms';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ValidationMessageService } from '../../core/validation-message.service';\r\nimport { FormGroupHelper } from '../../shared/form-group.helper';\r\nimport { FormControlBase } from '../form-control-base';\r\n\r\nexport type FileType = 'zip' | 'excel' | 'custom';\r\nexport type FileOutput = 'raw' | 'base64';\r\n\r\nexport const FileTypeExtensions = {\r\n  zip: ['.zip'],\r\n  excel: ['.xls', '.xlsx']\r\n};\r\n\r\n@Component({\r\n  selector: \"ec-file-upload\",\r\n  templateUrl: \"./file-upload.component.html\",\r\n  styleUrls: [\"./file-upload.component.scss\"]\r\n})\r\nexport class FileUploadComponent extends FormControlBase implements OnInit, OnChanges {\r\n\r\n  // static class to create the form group from a parent component\r\n  public static getFormModel(\r\n    validators: ValidatorFn[],\r\n    disabled: boolean = false\r\n  ): UntypedFormGroup {\r\n    let formGroup = new UntypedFormGroup({\r\n      file: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      name: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new UntypedFormControl(null),\r\n      uploadResult: new UntypedFormControl(null)\r\n    });\r\n    if (disabled) {\r\n      formGroup.disable();\r\n    }\r\n    return formGroup;\r\n  }\r\n\r\n  /**\r\n   * The value of the textbox input's placeholder\r\n   */\r\n  @Input() public placeholder?: string = \"Choose file...\";\r\n\r\n  /** Common extensions for a file browsing dialog\r\n   *  Note: Edge does not support the accept attribute on file inputs, therefor all file types\r\n   *        will be shown.  Firefox and Chrome both support this feature.\r\n   */\r\n  @Input() public fileType?: FileType | undefined;\r\n\r\n  /**\r\n   * File output, determines which properties are supplied on the formModel\r\n   */\r\n  @Input() public fileOutput?: FileOutput = 'base64';\r\n\r\n  /** If fileType is set to custom set the acceptable file types based on the custom array */\r\n  @Input() public customExtensions?: Array<string> | undefined;\r\n\r\n  /**\r\n   * Optional callback supported if the hosting page needs to process the file before\r\n   * setting the formModel with the file information. If the promise resolves it will continue\r\n   * and set the file name and contents to the formModel, otherwise on failure it'll do nothing.\r\n   */\r\n  @Input() public onFileSelected?: (file: File) => Promise<any>;\r\n\r\n  @ViewChild(\"fileInput\", { read: ElementRef, static: true }) public fileInput?: ElementRef;\r\n\r\n  /** Property bound to the file input to filter what file types are shown in the dialog */\r\n  public fileTypeAccept: string | undefined;\r\n\r\n  constructor(\r\n    protected validationMessageService: ValidationMessageService,\r\n    protected formGroupHelper: FormGroupHelper\r\n  ) {\r\n    super(validationMessageService, formGroupHelper);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges) {\r\n    super.ngOnChanges(changes);\r\n\r\n    this.updateFileTypeAccept();\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    super.ngOnInit();\r\n\r\n    // Watch for name to change, if the value is cleared we will clear the other\r\n    // supporting model properties. The name can be cleared by the user manually or via\r\n    // the clear button\r\n    this.formModel?.get('name')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          file: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n  }\r\n\r\n  public async fileChange(files: FileList): Promise<void> {\r\n    let file: File | null = files.item(0);\r\n\r\n    // If there is a file selected and then opened again and click cancel you get null so don't try and set anything\r\n    if (file) {\r\n      let reader: FileReader = new FileReader();\r\n\r\n      reader.onloadend = async e => {\r\n        let base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n        this.processFile(file!, base64FileString);\r\n      };\r\n\r\n      if (this.isBase64FileOutput()) {\r\n        reader.readAsDataURL(file);\r\n      } else {\r\n        await this.processFile(file!);\r\n      }\r\n\r\n      // Clear the file inputs value, this will allow the user to pick the same filename and cause\r\n      // the fileChange to trigger.\r\n      if (this.fileInput) {\r\n        this.fileInput.nativeElement.value = '';\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Checks the file type and updates the file type accept property. This is what determines the file\r\n   * type choices that the user will be limited to in the file browse dialog\r\n   */\r\n  private updateFileTypeAccept(): void {\r\n    if (this.fileType) {\r\n      if (this.fileType !== \"custom\") {\r\n        this.fileTypeAccept = FileTypeExtensions[this.fileType].join(\",\");\r\n      } else {\r\n        if (this.customExtensions) {\r\n          this.fileTypeAccept = this.customExtensions.join(\",\");\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Take a file that was selected by the user and process/patch our form model\r\n   * If the host component requires an action to occur with the file prior to the patch it will call\r\n   * and wait for it to return.\r\n   * @param file \r\n   * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64\r\n   */\r\n  private async processFile(file: File, base64FileString?: string | undefined): Promise<void> {\r\n    if (this.onFileSelected) {\r\n      try {\r\n        let result = await this.onFileSelected(file);\r\n        this.patchFileResult(file, base64FileString, result);\r\n      } catch (e) {\r\n        // Bummer, we're not going to do anything about it though. \r\n        // We are not patching any of the result so any existing information remains\r\n      }\r\n    } else {\r\n      this.patchFileResult(file, base64FileString);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Based on the fileOutput return whether this component is expected to deliver a base64 output\r\n   * @returns \r\n   */\r\n  private isBase64FileOutput(): boolean {\r\n    return this.fileOutput === 'base64';\r\n  }\r\n\r\n  /**\r\n   * When the file was selected and processed patch the file information that the hosting form will\r\n   * be looking for. \r\n   * @param file \r\n   * @param base64FileString \r\n   * @param onFileSelectedResult \r\n   */\r\n  private patchFileResult(file: File | null, base64FileString?: string, onFileSelectedResult?: any): void {\r\n    this.formModel?.patchValue({ \r\n      file: file, \r\n      name: file?.name, \r\n      base64FileString: base64FileString ?? null\r\n    });\r\n\r\n    if (onFileSelectedResult) {\r\n      this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n    } else {\r\n      this.formModel.patchValue({ uploadResult: null });\r\n    }\r\n  }\r\n}\r\n","<ec-form-group [label]=\"label\"\r\n               [formGroup]=\"formModel\"\r\n               class=\"mb-0\">\r\n  <div class=\"d-flex control-group\">\r\n    <div class=\"d-flex flex-grow position-relative\">\r\n      <input #fileInput\r\n             id=\"{{inputId}}_input\"\r\n             type=\"file\"\r\n             tabindex=\"-1\"\r\n             [attr.accept]=\"fileTypeAccept\"\r\n             (click)=\"browseBtn.focus()\"\r\n             (change)=\"fileChange($event.target.files)\"\r\n             [class.has-value]=\"formModel?.get('name')?.value\">\r\n      <ec-form-control id=\"{{inputId}}_formControl\"\r\n                       class=\"text-truncate\"\r\n                       [required]=\"required\"\r\n                       [pending]=\"pending\">\r\n        <input id=\"{{inputId}}_name\"\r\n               [formControl]=\"formModel?.get('name')\"\r\n               type=\"text\"\r\n               [placeholder]=\"placeholder\"\r\n               [tabindex]=\"-1\">\r\n      </ec-form-control>\r\n    </div>\r\n    <ec-button #browseBtn\r\n               id=\"{{inputId}}_browseBtn\"\r\n               (clicked)=\"fileInput.click()\"\r\n               type=\"secondary\"\r\n               [tabindex]=\"tabindex\"\r\n               [disabled]=\"formModel?.get('name')?.disabled\"\r\n               label=\"Browse\"\r\n               [autofocus]=\"autofocus\">\r\n    </ec-button>\r\n  </div>\r\n</ec-form-group>"]}
220
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.ts","../../../../../../projects/components/src/lib/controls/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, OnChanges, OnInit, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { UntypedFormControl, UntypedFormGroup, ValidatorFn } from '@angular/forms';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { ValidationMessageService } from '../../core/validation-message.service';\r\nimport { FormGroupHelper } from '../../shared/form-group.helper';\r\nimport { FormControlBase } from '../form-control-base';\r\n\r\nexport type FileType = 'zip' | 'excel' | 'custom';\r\nexport type FileOutput = 'raw' | 'base64';\r\n\r\nexport const FileTypeExtensions = {\r\n  zip: ['.zip'],\r\n  excel: ['.xls', '.xlsx']\r\n};\r\n\r\n@Component({\r\n  selector: \"ec-file-upload\",\r\n  templateUrl: \"./file-upload.component.html\",\r\n  styleUrls: [\"./file-upload.component.scss\"]\r\n})\r\nexport class FileUploadComponent extends FormControlBase implements OnInit, OnChanges {\r\n\r\n  // static class to create the form group from a parent component\r\n  public static getFormModel(\r\n    validators: ValidatorFn[],\r\n    disabled: boolean = false\r\n  ): UntypedFormGroup {\r\n    let formGroup = new UntypedFormGroup({\r\n      files: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      names: new UntypedFormControl({ value: null, disabled: disabled }, validators),\r\n      base64FileString: new UntypedFormControl(null),\r\n      uploadResult: new UntypedFormControl(null)\r\n    });\r\n    if (disabled) {\r\n      formGroup.disable();\r\n    }\r\n    return formGroup;\r\n  }\r\n\r\n  /**\r\n   * The value of the textbox input's placeholder\r\n   */\r\n  @Input() public placeholder?: string = \"Choose file...\";\r\n\r\n  /** Common extensions for a file browsing dialog\r\n   *  Note: Edge does not support the accept attribute on file inputs, therefor all file types\r\n   *        will be shown.  Firefox and Chrome both support this feature.\r\n   */\r\n  @Input() public fileType?: FileType | undefined;\r\n\r\n  /**\r\n   * File output, determines which properties are supplied on the formModel\r\n   */\r\n  @Input() public fileOutput?: FileOutput = 'base64';\r\n\r\n  /** If fileType is set to custom set the acceptable file types based on the custom array */\r\n  @Input() public customExtensions?: Array<string> | undefined;\r\n\r\n  /**\r\n   * Optional callback supported if the hosting page needs to process the file before\r\n   * setting the formModel with the file information. If the promise resolves it will continue\r\n   * and set the file name and contents to the formModel, otherwise on failure it'll do nothing.\r\n   */\r\n  @Input() public onFileSelected?: (files: File | File[]) => Promise<any>;\r\n\r\n  /** \r\n   * Optional display type that controls whether the file input textbox is displayed or\r\n   * simply a button the user clicks to launch the OS file storage dialog.\r\n   * Default: file\r\n   */\r\n  @Input() public displayType?: 'file' | 'button' = 'file';\r\n\r\n  /** \r\n   * When display type is set to button this property will control the button label\r\n   */\r\n  @Input() public buttonLabel?: string;\r\n\r\n  @ViewChild(\"fileInput\", { read: ElementRef, static: true }) public fileInput?: ElementRef;\r\n\r\n  /** Property bound to the file input to filter what file types are shown in the dialog */\r\n  public fileTypeAccept: string | undefined;\r\n\r\n  constructor(\r\n    protected validationMessageService: ValidationMessageService,\r\n    protected formGroupHelper: FormGroupHelper,\r\n  ) {\r\n    super(validationMessageService, formGroupHelper);\r\n  }\r\n\r\n  public ngOnChanges(changes: SimpleChanges) {\r\n    super.ngOnChanges(changes);\r\n\r\n    this.updateFileTypeAccept();\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    super.ngOnInit();\r\n\r\n    // Watch for name to change, if the value is cleared we will clear the other\r\n    // supporting model properties. The name can be cleared by the user manually or via\r\n    // the clear button\r\n    this.formModel?.get('name')?.valueChanges.pipe(\r\n      takeUntil(this.componentDestroyed)\r\n    ).subscribe(value => {\r\n      if (!value) {\r\n        this.formModel.patchValue({\r\n          files: null,\r\n          names: null,\r\n          base64FileString: null,\r\n          uploadResult: null\r\n        });\r\n      }\r\n    });\r\n  }\r\n\r\n  // * New fileChange method\r\n  public async fileChange(files: FileList): Promise<void> {\r\n    const promises = Array.from(files).map((file) => new Promise ((resolve) => {\r\n      if (file) {\r\n        const reader: FileReader = new FileReader();\r\n\r\n        reader.onloadend = async e => {\r\n          const base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n          await this.processFile(file, base64FileString);\r\n          resolve(null);\r\n        };\r\n\r\n        if (this.isBase64FileOutput()) {\r\n          reader.readAsDataURL(file);\r\n        } else {\r\n          resolve(this.processFile(file));\r\n        };\r\n      }\r\n    }));\r\n\r\n    await Promise.all(promises);\r\n\r\n    // Clear the file inputs value, this will allow the user to pick the same filenames again\r\n    // and cause fileChange to re-trigger.\r\n    if (this.fileInput) {\r\n      this.fileInput.nativeElement.value = '';\r\n    }\r\n  }\r\n\r\n  // * Old Method\r\n  // public async fileChange(files: FileList): Promise<void> {\r\n  //   let file = files.item(0);\r\n\r\n  //   // If there is a file selected and then opened again and click cancel you get null so don't try and set anything\r\n  //   if (file) {\r\n  //     let reader: FileReader = new FileReader();\r\n\r\n  //     reader.onloadend = async e => {\r\n  //       let base64FileString: string | undefined = reader?.result?.toString().split(\",\")[1];\r\n  //       this.processFile(file!, base64FileString);\r\n  //     };\r\n\r\n  //     if (this.isBase64FileOutput()) {\r\n  //       reader.readAsDataURL(file);\r\n  //     } else {\r\n  //       await this.processFile(file!);\r\n  //     }\r\n\r\n  //     // Clear the file inputs value, this will allow the user to pick the same filename and cause\r\n  //     // the fileChange to trigger.\r\n  //     if (this.fileInput) {\r\n  //       this.fileInput.nativeElement.value = '';\r\n  //     }\r\n  //   }\r\n  // }\r\n\r\n  /**\r\n   * Checks the file type and updates the file type accept property. This is what determines the file\r\n   * type choices that the user will be limited to in the file browse dialog\r\n   */\r\n  private updateFileTypeAccept(): void {\r\n    if (this.fileType) {\r\n      if (this.fileType !== \"custom\") {\r\n        this.fileTypeAccept = FileTypeExtensions[this.fileType].join(\",\");\r\n      } else {\r\n        if (this.customExtensions) {\r\n          this.fileTypeAccept = this.customExtensions.join(\",\");\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Take a file that was selected by the user and process/patch our form model\r\n   * If the host component requires an action to occur with the file prior to the patch it will call\r\n   * and wait for it to return.\r\n   * @param file \r\n   * @param base64FileString Optional: Will have a value provided if the fileOutput is set to base64\r\n   */\r\n  private async processFile(file: File, base64FileString?: string | undefined): Promise<void> {\r\n    if (this.onFileSelected) {\r\n      try {\r\n        let result = await this.onFileSelected(file);\r\n        this.patchFileResult(file, base64FileString, result);\r\n      } catch (e) {\r\n        // Bummer, we're not going to do anything about it though. \r\n        // We are not patching any of the result so any existing information remains\r\n      }\r\n    } else {\r\n      this.patchFileResult(file, base64FileString);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Based on the fileOutput return whether this component is expected to deliver a base64 output\r\n   * @returns \r\n   */\r\n  private isBase64FileOutput(): boolean {\r\n    return this.fileOutput === 'base64';\r\n  }\r\n\r\n  /**\r\n   * When the file was selected and processed patch the file information that the hosting form will\r\n   * be looking for. \r\n   * @param file \r\n   * @param base64FileString \r\n   * @param onFileSelectedResult \r\n   */\r\n  // * Old method\r\n  private patchFileResult(file: File | null, base64FileString?: string, onFileSelectedResult?: any): void {\r\n    // Get the current value of files from formModel\r\n    let files = this.formModel?.get('files')?.value || [];\r\n    let names = this.formModel?.get('names')?.value || [];\r\n\r\n    // If the file is not null, append it to the files array\r\n    if (file) {\r\n      files = [...files, file];\r\n    }\r\n\r\n    // If the file name is not null, append it to the names array\r\n    if (file?.name) {\r\n      names = [...names, file.name];\r\n    }\r\n\r\n    this.formModel?.patchValue({\r\n      files: files,\r\n      names: names,\r\n      base64FileString: base64FileString ?? null\r\n    });\r\n\r\n    if (onFileSelectedResult) {\r\n      this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n    } else {\r\n      this.formModel.patchValue({ uploadResult: null });\r\n    }\r\n\r\n  }\r\n\r\n  // * New method\r\n  // private patchFileResult(files: File[] | null, base64FileString?: string[], onFileSelectedResult?: any): void {\r\n  //   const fileArray = files || [];\r\n  //   const nameArray = fileArray.map(file => file.name);\r\n  //   const base64Array = base64FileString || [];\r\n\r\n  //   this.formModel?.patchValue({\r\n  //     files: fileArray,\r\n  //     name: nameArray,\r\n  //     base64FileString: base64Array\r\n  //   });\r\n    \r\n  //   if(onFileSelectedResult) {\r\n  //     this.formModel.patchValue({ uploadResult: onFileSelectedResult });\r\n  //   } else {\r\n  //     this.formModel.patchValue({ uploadResult: null });\r\n  //   };\r\n  // }\r\n}\r\n","<ec-form-group [label]=\"label\"\r\n               [formGroup]=\"formModel\"\r\n               class=\"mb-0\">\r\n  <div class=\"d-flex control-group\">\r\n    <div class=\"d-flex flex-grow position-relative\">\r\n      <input #fileInput\r\n             id=\"{{inputId}}_input\"\r\n             type=\"file\"\r\n             tabindex=\"-1\"\r\n             [attr.accept]=\"fileTypeAccept\"\r\n             (change)=\"fileChange($event.target.files)\"\r\n             [class.has-value]=\"formModel?.get('name')?.value\"\r\n             multiple>\r\n      <ec-form-control *ngIf=\"displayType === 'file'\"\r\n                       id=\"{{inputId}}_formControl\"\r\n                       class=\"text-truncate\"\r\n                       [required]=\"required\"\r\n                       [pending]=\"pending\">\r\n        <input id=\"{{inputId}}_name\"\r\n               [formControl]=\"formModel?.get('name')\"\r\n               type=\"text\"\r\n               [placeholder]=\"placeholder\"\r\n               [tabindex]=\"-1\">\r\n      </ec-form-control>\r\n    </div>\r\n    <ec-button *ngIf=\"displayType === 'file'\"\r\n               #browseBtn\r\n               id=\"{{inputId}}_browseBtn\"\r\n               (clicked)=\"fileInput.click()\"\r\n               type=\"secondary\"\r\n               [tabindex]=\"tabindex\"\r\n               [disabled]=\"formModel?.get('name')?.disabled\"\r\n               label=\"Browse\"\r\n               [autofocus]=\"autofocus\">\r\n    </ec-button>\r\n  </div>\r\n  <ec-button *ngIf=\"displayType === 'button'\"\r\n             #selectZipFilesBtn\r\n             id=\"{{inputId}}_browseBtn\"\r\n             [pending]=\"pending\"\r\n             type=\"primary\"\r\n             [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n             (clicked)=\"fileInput.click()\"\r\n             style=\"width: 100%;\">\r\n  </ec-button>\r\n</ec-form-group>"]}
@@ -3780,8 +3780,8 @@ class FileUploadComponent extends FormControlBase {
3780
3780
  // static class to create the form group from a parent component
3781
3781
  static getFormModel(validators, disabled = false) {
3782
3782
  let formGroup = new UntypedFormGroup({
3783
- file: new UntypedFormControl({ value: null, disabled: disabled }, validators),
3784
- name: new UntypedFormControl({ value: null, disabled: disabled }, validators),
3783
+ files: new UntypedFormControl({ value: null, disabled: disabled }, validators),
3784
+ names: new UntypedFormControl({ value: null, disabled: disabled }, validators),
3785
3785
  base64FileString: new UntypedFormControl(null),
3786
3786
  uploadResult: new UntypedFormControl(null)
3787
3787
  });
@@ -3802,6 +3802,12 @@ class FileUploadComponent extends FormControlBase {
3802
3802
  * File output, determines which properties are supplied on the formModel
3803
3803
  */
3804
3804
  this.fileOutput = 'base64';
3805
+ /**
3806
+ * Optional display type that controls whether the file input textbox is displayed or
3807
+ * simply a button the user clicks to launch the OS file storage dialog.
3808
+ * Default: file
3809
+ */
3810
+ this.displayType = 'file';
3805
3811
  }
3806
3812
  ngOnChanges(changes) {
3807
3813
  super.ngOnChanges(changes);
@@ -3816,38 +3822,65 @@ class FileUploadComponent extends FormControlBase {
3816
3822
  (_b = (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.get('name')) === null || _b === void 0 ? void 0 : _b.valueChanges.pipe(takeUntil(this.componentDestroyed)).subscribe(value => {
3817
3823
  if (!value) {
3818
3824
  this.formModel.patchValue({
3819
- file: null,
3825
+ files: null,
3826
+ names: null,
3820
3827
  base64FileString: null,
3821
3828
  uploadResult: null
3822
3829
  });
3823
3830
  }
3824
3831
  });
3825
3832
  }
3833
+ // * New fileChange method
3826
3834
  fileChange(files) {
3827
3835
  return __awaiter(this, void 0, void 0, function* () {
3828
- let file = files.item(0);
3829
- // If there is a file selected and then opened again and click cancel you get null so don't try and set anything
3830
- if (file) {
3831
- let reader = new FileReader();
3832
- reader.onloadend = (e) => __awaiter(this, void 0, void 0, function* () {
3833
- var _a;
3834
- let base64FileString = (_a = reader === null || reader === void 0 ? void 0 : reader.result) === null || _a === void 0 ? void 0 : _a.toString().split(",")[1];
3835
- this.processFile(file, base64FileString);
3836
- });
3837
- if (this.isBase64FileOutput()) {
3838
- reader.readAsDataURL(file);
3839
- }
3840
- else {
3841
- yield this.processFile(file);
3842
- }
3843
- // Clear the file inputs value, this will allow the user to pick the same filename and cause
3844
- // the fileChange to trigger.
3845
- if (this.fileInput) {
3846
- this.fileInput.nativeElement.value = '';
3836
+ const promises = Array.from(files).map((file) => new Promise((resolve) => {
3837
+ if (file) {
3838
+ const reader = new FileReader();
3839
+ reader.onloadend = (e) => __awaiter(this, void 0, void 0, function* () {
3840
+ var _a;
3841
+ const base64FileString = (_a = reader === null || reader === void 0 ? void 0 : reader.result) === null || _a === void 0 ? void 0 : _a.toString().split(",")[1];
3842
+ yield this.processFile(file, base64FileString);
3843
+ resolve(null);
3844
+ });
3845
+ if (this.isBase64FileOutput()) {
3846
+ reader.readAsDataURL(file);
3847
+ }
3848
+ else {
3849
+ resolve(this.processFile(file));
3850
+ }
3851
+ ;
3847
3852
  }
3853
+ }));
3854
+ yield Promise.all(promises);
3855
+ // Clear the file inputs value, this will allow the user to pick the same filenames again
3856
+ // and cause fileChange to re-trigger.
3857
+ if (this.fileInput) {
3858
+ this.fileInput.nativeElement.value = '';
3848
3859
  }
3849
3860
  });
3850
3861
  }
3862
+ // * Old Method
3863
+ // public async fileChange(files: FileList): Promise<void> {
3864
+ // let file = files.item(0);
3865
+ // // If there is a file selected and then opened again and click cancel you get null so don't try and set anything
3866
+ // if (file) {
3867
+ // let reader: FileReader = new FileReader();
3868
+ // reader.onloadend = async e => {
3869
+ // let base64FileString: string | undefined = reader?.result?.toString().split(",")[1];
3870
+ // this.processFile(file!, base64FileString);
3871
+ // };
3872
+ // if (this.isBase64FileOutput()) {
3873
+ // reader.readAsDataURL(file);
3874
+ // } else {
3875
+ // await this.processFile(file!);
3876
+ // }
3877
+ // // Clear the file inputs value, this will allow the user to pick the same filename and cause
3878
+ // // the fileChange to trigger.
3879
+ // if (this.fileInput) {
3880
+ // this.fileInput.nativeElement.value = '';
3881
+ // }
3882
+ // }
3883
+ // }
3851
3884
  /**
3852
3885
  * Checks the file type and updates the file type accept property. This is what determines the file
3853
3886
  * type choices that the user will be limited to in the file browse dialog
@@ -3902,11 +3935,23 @@ class FileUploadComponent extends FormControlBase {
3902
3935
  * @param base64FileString
3903
3936
  * @param onFileSelectedResult
3904
3937
  */
3938
+ // * Old method
3905
3939
  patchFileResult(file, base64FileString, onFileSelectedResult) {
3906
- var _a;
3907
- (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.patchValue({
3908
- file: file,
3909
- name: file === null || file === void 0 ? void 0 : file.name,
3940
+ var _a, _b, _c, _d, _e;
3941
+ // Get the current value of files from formModel
3942
+ let files = ((_b = (_a = this.formModel) === null || _a === void 0 ? void 0 : _a.get('files')) === null || _b === void 0 ? void 0 : _b.value) || [];
3943
+ let names = ((_d = (_c = this.formModel) === null || _c === void 0 ? void 0 : _c.get('names')) === null || _d === void 0 ? void 0 : _d.value) || [];
3944
+ // If the file is not null, append it to the files array
3945
+ if (file) {
3946
+ files = [...files, file];
3947
+ }
3948
+ // If the file name is not null, append it to the names array
3949
+ if (file === null || file === void 0 ? void 0 : file.name) {
3950
+ names = [...names, file.name];
3951
+ }
3952
+ (_e = this.formModel) === null || _e === void 0 ? void 0 : _e.patchValue({
3953
+ files: files,
3954
+ names: names,
3910
3955
  base64FileString: base64FileString !== null && base64FileString !== void 0 ? base64FileString : null
3911
3956
  });
3912
3957
  if (onFileSelectedResult) {
@@ -3918,10 +3963,10 @@ class FileUploadComponent extends FormControlBase {
3918
3963
  }
3919
3964
  }
3920
3965
  FileUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, deps: [{ token: ValidationMessageService }, { token: FormGroupHelper }], target: i0.ɵɵFactoryTarget.Component });
3921
- FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (click)=\"browseBtn.focus()\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\">\r\n <ec-form-control id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }] });
3966
+ FileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FileUploadComponent, selector: "ec-file-upload", inputs: { placeholder: "placeholder", fileType: "fileType", fileOutput: "fileOutput", customExtensions: "customExtensions", onFileSelected: "onFileSelected", displayType: "displayType", buttonLabel: "buttonLabel" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true, read: ElementRef, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: ButtonComponent, selector: "ec-button", inputs: ["id", "disabled", "icon", "label", "badge", "tabindex", "type", "pending", "pendingIcon", "customTemplate", "isSubmit", "autofocus"], outputs: ["clicked"] }, { kind: "component", type: FormControlComponent, selector: "ec-form-control", inputs: ["id", "icon", "actionIcon", "showClear", "pending", "required", "readonly"], outputs: ["actionClicked"] }, { kind: "component", type: FormGroupComponent, selector: "ec-form-group", inputs: ["id", "label", "formGroup", "labelPosition", "overrideValidationError", "hideValidationMessage"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
3922
3967
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FileUploadComponent, decorators: [{
3923
3968
  type: Component,
3924
- args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (click)=\"browseBtn.focus()\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\">\r\n <ec-form-control id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
3969
+ args: [{ selector: "ec-file-upload", template: "<ec-form-group [label]=\"label\"\r\n [formGroup]=\"formModel\"\r\n class=\"mb-0\">\r\n <div class=\"d-flex control-group\">\r\n <div class=\"d-flex flex-grow position-relative\">\r\n <input #fileInput\r\n id=\"{{inputId}}_input\"\r\n type=\"file\"\r\n tabindex=\"-1\"\r\n [attr.accept]=\"fileTypeAccept\"\r\n (change)=\"fileChange($event.target.files)\"\r\n [class.has-value]=\"formModel?.get('name')?.value\"\r\n multiple>\r\n <ec-form-control *ngIf=\"displayType === 'file'\"\r\n id=\"{{inputId}}_formControl\"\r\n class=\"text-truncate\"\r\n [required]=\"required\"\r\n [pending]=\"pending\">\r\n <input id=\"{{inputId}}_name\"\r\n [formControl]=\"formModel?.get('name')\"\r\n type=\"text\"\r\n [placeholder]=\"placeholder\"\r\n [tabindex]=\"-1\">\r\n </ec-form-control>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'file'\"\r\n #browseBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n (clicked)=\"fileInput.click()\"\r\n type=\"secondary\"\r\n [tabindex]=\"tabindex\"\r\n [disabled]=\"formModel?.get('name')?.disabled\"\r\n label=\"Browse\"\r\n [autofocus]=\"autofocus\">\r\n </ec-button>\r\n </div>\r\n <ec-button *ngIf=\"displayType === 'button'\"\r\n #selectZipFilesBtn\r\n id=\"{{inputId}}_browseBtn\"\r\n [pending]=\"pending\"\r\n type=\"primary\"\r\n [label]=\"buttonLabel ?? 'Browse_TC' | translate\"\r\n (clicked)=\"fileInput.click()\"\r\n style=\"width: 100%;\">\r\n </ec-button>\r\n</ec-form-group>", styles: [":host{display:block;margin-bottom:1rem}ec-form-control{margin-bottom:0}ec-form-control ::ng-deep>.ec-focus-ring{display:none!important}input[type=file]{opacity:0;display:block;position:absolute;top:0;left:0;width:100%;height:100%;z-index:1;cursor:pointer}input[type=file].has-value{width:calc(100% - 1.5rem)}ec-button{--ec-button-border-color-secondary: var(--ec-form-control-border-color);--ec-button-color-icon-secondary: var(--ec-color-icon)}\n"] }]
3925
3970
  }], ctorParameters: function () { return [{ type: ValidationMessageService }, { type: FormGroupHelper }]; }, propDecorators: { placeholder: [{
3926
3971
  type: Input
3927
3972
  }], fileType: [{
@@ -3932,6 +3977,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
3932
3977
  type: Input
3933
3978
  }], onFileSelected: [{
3934
3979
  type: Input
3980
+ }], displayType: [{
3981
+ type: Input
3982
+ }], buttonLabel: [{
3983
+ type: Input
3935
3984
  }], fileInput: [{
3936
3985
  type: ViewChild,
3937
3986
  args: ["fileInput", { read: ElementRef, static: true }]